$(function () {
  $.fn.extend({
    fixedThead: function (options) {
      var _that = $(this);
      var option = {
        height: 400,
        shadow: true,
        resize: true
      };
      options = $.extend(option, options);
      if ($(this).find('table').length === 0) {
        return false;
      }
      var _height = $(this)[0].style.height, _table_config = _height.match(/([0-9]+)([%\w]+)/);
      if (_table_config === null) {
        _table_config = [null, options.height, 'px'];
      } else {
        $(this).css({
          'boxSizing': 'content-box',
          'paddingBottom': $(this).find('thead').height()
        });
      }
      $(this).css({ 'position': 'relative' });
      var _thead = $(this).find('thead')[0].outerHTML,
        _tbody = $(this).find('tbody')[0].outerHTML,
        _thead_div = $('<div class="thead_div"><table class="table table-hover mb0"></table></div>'),
        _shadow_top = $('<div class="tbody_shadow_top"></div>'),
        _tbody_div = $('<div class="tbody_div" style="height:' + _table_config[1] + _table_config[2] + ';"><table class="table table-hover mb0" style="margin-top:-' + $(this).find('thead').height() + 'px"></table></div>'),
        _shadow_bottom = $('<div class="tbody_shadow_bottom"></div>');
      _thead_div.find('table').append(_thead);
      _tbody_div.find('table').append(_thead);
      _tbody_div.find('table').append(_tbody);
      $(this).html('');
      $(this).append(_thead_div);
      $(this).append(_shadow_top);
      $(this).append(_tbody_div);
      $(this).append(_shadow_bottom);
      var _table_width = _that.find('.thead_div table')[0].offsetWidth,
        _body_width = _that.find('.tbody_div table')[0].offsetWidth,
        _length = _that.find('tbody tr:eq(0)>td').length;
      $(this).find('tbody tr:eq(0)>td').each(function (index, item) {
        var _item = _that.find('thead tr:eq(0)>th').eq(index);
        if (index === (_length - 1)) {
          _item.attr('width', $(item)[0].clientWidth + (_table_width - _body_width));
        } else {
          _item.attr('width', $(item)[0].offsetWidth);
        }
      });
      if (options.resize) {
        $(window).resize(function () {
          var _table_width = _that.find('.thead_div table')[0].offsetWidth,
            _body_width = _that.find('.tbody_div table')[0].offsetWidth,
            _length = _that.find('tbody tr:eq(0)>td').length;
          _that.find('tbody tr:eq(0)>td').each(function (index, item) {
            var _item = _that.find('thead tr:eq(0)>th').eq(index);
            if (index === (_length - 1)) {
              _item.attr('width', $(item)[0].clientWidth + (_table_width - _body_width));
            } else {
              _item.attr('width', $(item)[0].offsetWidth);
            }
          });
        });
      }
      if (options.shadow) {
        var table_body = $(this).find('.tbody_div')[0];
        if (_table_config[1] >= table_body.scrollHeight) {
          $(this).find('.tbody_shadow_top').hide();
          $(this).find('.tbody_shadow_bottom').hide();
        } else {
          $(this).find('.tbody_shadow_top').hide();
          $(this).find('.tbody_shadow_bottom').show();
        }
        $(this).find('.tbody_div').scroll(function (e) {
          var _scrollTop = $(this)[0].scrollTop,
            _scrollHeight = $(this)[0].scrollHeight,
            _clientHeight = $(this)[0].clientHeight,
            _shadow_top = _that.find('.tbody_shadow_top'),
            _shadow_bottom = _that.find('.tbody_shadow_bottom');
          if (_scrollTop == 0) {
            _shadow_top.hide();
            _shadow_bottom.show();
          } else if (_scrollTop > 0 && _scrollTop < (_scrollHeight - _clientHeight)) {
            _shadow_top.show();
            _shadow_bottom.show();
          } else if (_scrollTop == (_scrollHeight - _clientHeight)) {
            _shadow_top.show();
            _shadow_bottom.hide();
          }
        })
      }
    }

  });

}(jQuery))

$(document).ready(function () {
  $(".sub-menu a.sub-menu-a").click(function () {
    $(this).next(".sub").slideToggle("slow").siblings(".sub:visible").slideUp("slow");
  });
});
var aceEditor = {
  layer_view: '',
  aceConfig: {},  //ace配置参数
  editor: null,
  pathAarry: [],
  editorLength: 0,
  isAceView: true,
  ace_active: '',
  is_resizing: false,
  menu_path: '', //当前文件目录根地址
  refresh_config: {
    el: {}, // 需要重新获取的元素,为DOM对象
    path: '',// 需要获取的路径文件信息
    group: 1,// 当前列表层级，用来css固定结构
    is_empty: true
  }, //刷新配置参数
  // 事件编辑器-方法，事件绑定
  eventEditor: function () {
    var _this = this, _icon = '<span class="icon"><i class="glyphicon glyphicon-ok" aria-hidden="true"></i></span>';
    $(window).resize(function () {
      if (_this.ace_active != undefined) _this.setEditorView()
      if ($('.aceEditors .layui-layer-maxmin').length > 0) {
        $('.aceEditors').css({
          'top': 0,
          'left': 0,
          'width': $(this)[0].innerWidth,
          'height': $(this)[0].innerHeight
        });
      }
    })
    $(document).click(function (e) {
      $('.ace_toolbar_menu').hide();
      $('.ace_conter_editor .ace_editors').css('fontSize', _this.aceConfig.aceEditor.fontSize + 'px');
      $('.ace_toolbar_menu .menu-tabs,.ace_toolbar_menu .menu-encoding,.ace_toolbar_menu .menu-files').hide();
    });
    $('.ace_editor_main').on('click', function () {
      $('.ace_toolbar_menu').hide();
    });
    $('.ace_toolbar_menu').click(function (e) {
      e.stopPropagation();
      e.preventDefault();
    });
    // 显示工具条
    $('.ace_header .pull-down').click(function () {
      if ($(this).find('i').hasClass('glyphicon-menu-down')) {
        $('.ace_header').css({ 'top': '-35px' });
        $('.ace_overall').css({ 'top': '0' });
        $(this).css({ 'top': '35px', 'height': '40px', 'line-height': '40px' });
        $(this).find('i').addClass('glyphicon-menu-up').removeClass('glyphicon-menu-down');
      } else {
        $('.ace_header').css({ 'top': '0' });
        $('.ace_overall').css({ 'top': '35px' });
        $(this).removeAttr('style');
        $(this).find('i').addClass('glyphicon-menu-down').removeClass('glyphicon-menu-up');
      }
      _this.setEditorView();
    });
    // 切换TAB视图
    $('.ace_conter_menu').on('click', '.item', function (e) {
      var _id = $(this).attr('data-id'), _item = _this.editor[_id]
      $('.item_tab_' + _id).addClass('active').siblings().removeClass('active');
      $('#ace_editor_' + _id).addClass('active').siblings().removeClass('active');
      _this.ace_active = _id;
      _this.currentStatusBar(_id);
      _this.is_file_history(_item);
    });
    // 移上TAB按钮变化，仅文件被修改后
    $('.ace_conter_menu').on('mouseover', '.item .icon-tool', function () {
      var type = $(this).attr('data-file-state');
      if (type != '0') {
        $(this).removeClass('glyphicon-exclamation-sign').addClass('glyphicon-remove');
      }
    });
    // 移出tab按钮变化，仅文件被修改后
    $('.ace_conter_menu').on('mouseout', '.item .icon-tool', function () {
      var type = $(this).attr('data-file-state');
      if (type != '0') {
        $(this).removeClass('glyphicon-remove').addClass('glyphicon-exclamation-sign');
      }
    });
    // 关闭编辑视图
    $('.ace_conter_menu').on('click', '.item .icon-tool', function (e) {
      var file_type = $(this).attr('data-file-state');
      var file_title = $(this).attr('data-title');
      var _id = $(this).parent().parent().attr('data-id');
      switch (file_type) {
        // 直接关闭
        case '0':
          _this.removeEditor(_id);
          break;
        // 未保存
        case '1':
          var loadT = layer.open({
            type: 1,
            area: ['400px', '180px'],
            title: '提示',
            content: '<div class="ace-clear-form">\
							<div class="clear-icon"></div>\
							<div class="clear-title">是否保存对&nbsp<span class="size_ellipsis" style="max-width:150px;vertical-align: top;" title="' + file_title + '">' + file_title + '</span>&nbsp的更改？</div>\
							<div class="clear-tips">如果不保存，更改会丢失！</div>\
							<div class="ace-clear-btn" style="">\
								<button type="button" class="btn btn-sm btn-default" style="float:left" data-type="2">不保存文件</button>\
								<button type="button" class="btn btn-sm btn-default" style="margin-right:10px;" data-type="1">取消</button>\
								<button type="button" class="btn btn-sm btn-success" data-type="0">保存文件</button>\
							</div>\
						</div>',
            success: function (layers, index) {
              $('.ace-clear-btn .btn').click(function () {
                var _type = $(this).attr('data-type'),
                  _item = _this.editor[_id];
                switch (_type) {
                  case '0': //保存文件
                    _this.saveFileMethod(_item);
                    break;
                  case '1': //关闭视图
                    layer.close(index);
                    break;
                  case '2': //取消保存
                    _this.removeEditor(_id);
                    layer.close(index);
                    break;
                }
              });
            }
          });
          break;
      }
      $('.ace_toolbar_menu').hide();
      $('.ace_toolbar_menu .menu-tabs,.ace_toolbar_menu .menu-encoding,.ace_toolbar_menu .menu-files').hide();
      e.stopPropagation();
      e.preventDefault();
    });
    $(window).keyup(function (e) {
      if (e.keyCode === 116 && $('#ace_conter').length == 1) {
        layer.msg('编辑器模式下无法刷新网页，请关闭后重试');
      }
    });
    // 新建编辑器视图
    $('.ace_editor_add').click(function () {
      _this.addEditorView();
    });
    // 底部状态栏功能按钮
    $('.ace_conter_toolbar .pull-right span').click(function (e) {
      var _type = $(this).attr('data-type'),
        _item = _this.editor[_this.ace_active];
      $('.ace_toolbar_menu').show();
      switch (_type) {
        case 'cursor':
          $('.ace_toolbar_menu').hide();
          $('.ace_header .jumpLine').click();
          break;

        case 'history':
          $('.ace_toolbar_menu').hide();
          if (_item.historys.length === 0) {
            layer.msg('历史文件为空', { icon: 0 });
            return false;
          }
          _this.layer_view = layer.open({
            type: 1,
            area: '550px',
            title: '文件历史版本[ ' + _item.fileName + ' ]',
            skin: 'historys_layer',
            content: '<div class="pd20">\
							<div class="divtable">\
								<table class="historys table table-hover">\
									<thead><tr><th>文件名</th><th>版本时间</th><th style="text-align:right;">操作</th></tr></thead>\
									<tbody></tbody>\
								</table>\
							</div>\
						</div>',
            success: function (layeo, index) {
              var _html = '';
              for (var i = 0; i < _item.historys.length; i++) {
                _html += '<tr><td><span class="size_ellipsis" style="max-width:200px">' + _item.fileName + '</span></td><td>' + bt.format_data(_item.historys[i]) + '</td><td align="right"><a href="javascript:;" class="btlink open_history_file" data-time="' + _item.historys[i] + '">打开文件</a>&nbsp;&nbsp;|&nbsp;&nbsp;<a href="javascript:;" class="btlink recovery_file_historys" data-history="' + _item.historys[i] + '" data-path="' + _item.path + '">恢复</a></td></tr>'
              }
              if (_html === '') _html += '<tr><td colspan="3">当前文件无历史版本</td></tr>'
              $('.historys tbody').html(_html);
              $('.historys_layer').css('top', ($(window).height() / 2) - ($('.historys_layer').height() / 2) + 'px')
              $('.open_history_file').click(function () {
                var _history = $(this).attr('data-time');
                _this.openHistoryEditorView({ filename: _item.path, history: _history }, function () {
                  layer.close(index);
                  $('.ace_conter_tips').show();
                  $('.ace_conter_tips .tips').html('只读文件，文件为' + _item.path + '，历史版本 [ ' + bt.format_data(new Number(_history)) + ' ]<a href="javascript:;" class="ml35 btlink" data-path="' + _item.path + '" data-history="' + _history + '">点击恢复当前历史版本</a>');
                });
              });
              $('.recovery_file_historys').click(function () {
                _this.event_ecovery_file(this);
              });
            }
          });
          break;
        case 'tab':
          $('.ace_toolbar_menu .menu-tabs').show().siblings().hide();
          $('.tabsType').find(_item.softTabs ? '[data-value="nbsp"]' : '[data-value="tabs"]').addClass('active').append(_icon);
          $('.tabsSize [data-value="' + _item.tabSize + '"]').addClass('active').append(_icon);
          break;
        case 'encoding':
          _this.getEncodingList(_item.encoding);
          $('.ace_toolbar_menu .menu-encoding').show().siblings().hide();
          break;
        case 'lang':
          $('.ace_toolbar_menu').hide();
          layer.msg('暂不支持切换语言模式，敬请期待!', { icon: 6 });
          break;
      }
      e.stopPropagation();
      e.preventDefault();
    });
    // 隐藏目录
    $('.tips_fold_icon .glyphicon').click(function () {
      if ($(this).hasClass('glyphicon-menu-left')) {
        $('.ace_conter_tips').css('right', '0');
        $('.tips_fold_icon').css('left', '0');
        $(this).removeClass('glyphicon-menu-left').addClass('glyphicon-menu-right');
      } else {
        $('.ace_conter_tips').css('right', '-100%');
        $('.tips_fold_icon').css('left', '-25px');
        $(this).removeClass('glyphicon-menu-right').addClass('glyphicon-menu-left');
      }
    });
    // 设置换行符
    $('.menu-tabs').on('click', 'li', function (e) {
      var _val = $(this).attr('data-value'), _item = _this.editor[_this.ace_active];
      if ($(this).parent().hasClass('tabsType')) {
        _item.ace.getSession().setUseSoftTabs(_val == 'nbsp');
        _item.softTabs = _val == 'nbsp';
      } else {
        _item.ace.getSession().setTabSize(_val);
        _item.tabSize = _val;
      }
      $(this).siblings().removeClass('active').find('.icon').remove();
      $(this).addClass('active').append(_icon);
      _this.currentStatusBar(_item.id);
      e.stopPropagation();
      e.preventDefault();
    });
    // 设置编码内容
    $('.menu-encoding').on('click', 'li', function (e) {
      var _item = _this.editor[_this.ace_active];
      layer.msg('设置文件编码：' + $(this).attr('data-value'));
      $('.ace_conter_toolbar [data-type="encoding"]').html('编码：<i>' + $(this).attr('data-value') + '</i>');
      $(this).addClass('active').append(_icon).siblings().removeClass('active').find('span').remove();
      _item.encoding = $(this).attr('data-value');
      _this.saveFileMethod(_item);
    });
    // 搜索内容键盘事件
    $('.menu-files .menu-input').keyup(function () {
      _this.searchRelevance($(this).val());
      if ($(this).val != '') {
        $(this).next().show();
      } else {
        $(this).next().hide();
      }
    });
    // 清除搜索内容事件
    $('.menu-files .menu-conter .fa').click(function () {
      $('.menu-files .menu-input').val('').next().hide();
      _this.searchRelevance();
    });
    // 顶部状态栏
    $('.ace_header>span').click(function (e) {
      var type = $(this).attr('class'), _item = _this.editor[_this.ace_active];
      if (_this.ace_active == '' && type != 'helps') {
        return false;
      }
      switch (type) {
        case 'saveFile': //保存当时文件
          _this.saveFileMethod(_item);
          break;
        case 'saveFileAll': //保存全部
          var loadT = layer.open({
            type: 1,
            area: ['350px', '180px'],
            title: '提示',
            content: '<div class="ace-clear-form">\
							<div class="clear-icon"></div>\
							<div class="clear-title">是否保存对全部文件的更改？</div>\
							<div class="clear-tips">如果不保存，更改会丢失！</div>\
							<div class="ace-clear-btn" style="">\
								<button type="button" class="btn btn-sm btn-default clear-btn" style="margin-right:10px;" >取消</button>\
								<button type="button" class="btn btn-sm btn-success save-all-btn">保存文件</button>\
							</div>\
						</div>',
            success: function (layers, index) {
              $('.clear-btn').click(function () {
                layer.close(index);
              });
              $('.save-all-btn').click(function () {
                var _arry = [], editor = aceEditor['editor'];
                for (var item in editor) {
                  _arry.push({
                    path: editor[item]['path'],
                    data: editor[item]['ace'].getValue(),
                    encoding: editor[item]['encoding'],
                  })
                }
                _this.saveAllFileBody(_arry, function () {
                  $('.ace_conter_menu>.item').each(function (el, index) {
                    $(this).find('i').attr('data-file-state', '0').removeClass('glyphicon-exclamation-sign').addClass('glyphicon-remove');
                    _item.fileType = 0;
                  });
                  layer.close(index);
                });
              });
            }
          });
          break;
        case 'refreshs': //刷新文件
          if (_item.fileType === 0) {
            aceEditor.getFileBody({ path: _item.path }, function (res) {
              _item.ace.setValue(res.data);
              _item.fileType = 0;
              $('.item_tab_' + _item.id + ' .icon-tool').attr('data-file-state', '0').removeClass('glyphicon-exclamation-sign').addClass('glyphicon-remove');
              layer.msg('刷新成功', { icon: 1 });
            });
            return false;
          }
          var loadT = layer.open({
            type: 1,
            area: ['350px', '180px'],
            title: '提示',
            content: '<div class="ace-clear-form">\
							<div class="clear-icon"></div>\
							<div class="clear-title">是否刷新当前文件</div>\
							<div class="clear-tips">刷新当前文件会覆盖当前修改,是否继续！</div>\
							<div class="ace-clear-btn" style="">\
								<button type="button" class="btn btn-sm btn-default clear-btn" style="margin-right:10px;" >取消</button>\
								<button type="button" class="btn btn-sm btn-success save-all-btn">确定</button>\
							</div>\
						</div>',
            success: function (layers, index) {
              $('.clear-btn').click(function () {
                layer.close(index);
              });
              $('.save-all-btn').click(function () {
                aceEditor.getFileBody({ path: _item.path }, function (res) {
                  layer.close(index);
                  _item.ace.setValue(res.data);
                  _item.fileType == 0;
                  $('.item_tab_' + _item.id + ' .icon-tool').attr('data-file-state', '0').removeClass('glyphicon-exclamation-sign').addClass('glyphicon-remove');
                  layer.msg('刷新成功', { icon: 1 });
                });
              });
            }
          });
          break;
        // 搜索
        case 'searchs':
          _item.ace.execCommand('find');
          break;
        // 替换
        case 'replaces':
          _item.ace.execCommand('replace');
          break;
        // 跳转行
        case 'jumpLine':
          $('.ace_toolbar_menu').show().find('.menu-jumpLine').show().siblings().hide();
          $('.set_jump_line input').val('').focus();
          var _cursor = aceEditor.editor[aceEditor.ace_active].ace.selection.getCursor();
          $('.set_jump_line .jump_tips span:eq(0)').text(_cursor.row);
          $('.set_jump_line .jump_tips span:eq(1)').text(_cursor.column);
          $('.set_jump_line .jump_tips span:eq(2)').text(aceEditor.editor[aceEditor.ace_active].ace.session.getLength());
          $('.set_jump_line input').unbind('keyup').on('keyup', function (e) {
            var _val = $(this).val();
            if ((e.keyCode >= 48 && e.keyCode <= 57) || (e.keyCode >= 96 && e.keyCode <= 105)) {
              if (_val != '' && typeof parseInt(_val) == 'number') {
                _item.ace.gotoLine(_val);
              };
            }
          });
          break;
        // 字体
        case 'fontSize':
          $('.ace_toolbar_menu').show().find('.menu-fontSize').show().siblings().hide();
          $('.menu-fontSize .set_font_size input').val(_this.aceConfig.aceEditor.fontSize).focus();
          $('.menu-fontSize set_font_size input').unbind('keypress onkeydown').on('keypress onkeydown', function (e) {
            var _val = $(this).val();
            if (_val == '') {
              $(this).css('border', '1px solid red');
              $(this).next('.tips').text('字体设置范围 12-45');
            } else if (!isNaN(_val)) {
              $(this).removeAttr('style');
              if (parseInt(_val) > 11 && parseInt(_val) < 45) {
                $('.ace_conter_editor .ace_editors').css('fontSize', _val + 'px')
              } else {
                $('.ace_conter_editor .ace_editors').css('fontSize', '13px');
                $(this).css('border', '1px solid red');
                $(this).next('.tips').text('字体设置范围 12-45');
              }
            } else {
              $(this).css('border', '1px solid red');
              $(this).next('.tips').text('字体设置范围 12-45');
            }
            e.stopPropagation();
            e.preventDefault();
          });
          $('.menu-fontSize .menu-conter .set_font_size input').unbind('change').change(function () {
            var _val = $(this).val();
            $('.ace_conter_editor .ace_editors').css('fontSize', _val + 'px');
          });
          $('.set_font_size .btn-save').unbind('click').click(function () {
            var _fontSize = $('.set_font_size input').val();
            _this.aceConfig.aceEditor.fontSize = parseInt(_fontSize);
            _this.saveAceConfig(_this.aceConfig, function (res) {
              if (res.status) {
                $('.ace_editors').css('fontSize', _fontSize + 'px');
                layer.msg('设置成功', { icon: 1 });
              }
            });
          });
          break;
        //主题
        case 'themes':
          $('.ace_toolbar_menu').show().find('.menu-themes').show().siblings().hide();
          var _html = '', _arry = ['白色主题', '黑色主题'];
          for (var i = 0; i < _this.aceConfig.themeList.length; i++) {
            if (_this.aceConfig.themeList[i] != _this.aceConfig.aceEditor.editorTheme) {
              _html += '<li data-value="' + _this.aceConfig.themeList[i] + '">' + _this.aceConfig.themeList[i] + '【' + _arry[i] + '】</li>';
            } else {
              _html += '<li data-value="' + _this.aceConfig.themeList[i] + '" class="active">' + _this.aceConfig.themeList[i] + '【' + _arry[i] + '】' + _icon + '</li>';
            }
          }
          $('.menu-themes ul').html(_html);
          $('.menu-themes ul li').click(function () {
            var _theme = $(this).attr('data-value');
            $(this).addClass('active').append(_icon).siblings().removeClass('active').find('.icon').remove();
            _this.aceConfig.aceEditor.editorTheme = _theme;
            _this.saveAceConfig(_this.aceConfig, function (res) {
              for (var item in _this.editor) {
                _this.editor[item].ace.setTheme("ace/theme/" + _theme);
              }
              layer.msg('设置成功', { icon: 1 });
            });
          });
          break;
        case 'setUp':
          $('.ace_toolbar_menu').show().find('.menu-setUp').show().siblings().hide();
          $('.menu-setUp .editor_menu li').each(function (index, el) {
            var _type = _this.aceConfig.aceEditor[$(el).attr('data-type')];
            if (_type) $(el).addClass('active').append(_icon);
          })
          $('.menu-setUp .editor_menu li').unbind('click').click(function () {
            var _type = $(this).attr('data-type');
            _this.aceConfig.aceEditor[_type] = !$(this).hasClass('active');
            if ($(this).hasClass('active')) {
              $(this).removeClass('active').find('.icon').remove();
            } else {
              $(this).addClass('active').append(_icon);
            }
            _this.saveAceConfig(_this.aceConfig, function (res) {
              for (var item in _this.editor) {
                _this.editor[item].ace.setOption(_type, _this.aceConfig.aceEditor[_type]);
              }
              layer.msg('设置成功', { icon: 1 });
            });
          });
          break;
        case 'helps':
          if (!$('[data-type=shortcutKeys]').length != 0) {
            _this.addEditorView(1, { title: '快捷键提示', html: aceShortcutKeys.innerHTML });
          } else {
            $('[data-type=shortcutKeys]').click();
          }
          break;
      }

      e.stopPropagation();
      e.preventDefault();
    });

    // 文件目录选择
    $('.ace_catalogue_list').on('click', '.has-children .file_fold', function (e) {
      var _layers = $(this).attr('data-layer'), _type = $(this).find('data-type'), _path = $(this).parent().attr('data-menu-path'), _menu = $(this).find('.glyphicon'), _group = parseInt($(this).attr('data-group')), _file = $(this).attr('data-file'), _tath = $(this);
      var _active = $('.ace_catalogue_list .has-children .file_fold.edit_file_group');
      if (_active.length > 0 && $(this).attr('data-edit') === undefined) {
        switch (_active.attr('data-edit')) {
          case '2':
            _active.find('.file_input').siblings().show();
            _active.find('.file_input').remove();
            _active.removeClass('edit_file_group').removeAttr('data-edit');
            break;
          case '1':
          case '0':
            _active.parent().remove();
            break;
        }
        layer.closeAll('tips');
      }
      $('.ace_toolbar_menu').hide();
      $('.ace_toolbar_menu .menu-tabs,.ace_toolbar_menu .menu-encoding,.ace_toolbar_menu .menu-files').hide();
      if ($(this).hasClass('edit_file_group')) return false;
      $('.ace_catalogue_list .has-children .file_fold').removeClass('bg');
      $(this).addClass('bg');
      if ($(this).attr('data-file') == 'Dir') {
        if (_menu.hasClass('glyphicon-menu-right')) {
          _menu.removeClass('glyphicon-menu-right').addClass('glyphicon-menu-down');
          $(this).next().show();
          if ($(this).next().find('li').length == 0) _this.reader_file_dir_menu({ el: $(this).next(), path: _path, group: _group + 1 });
        } else {
          _menu.removeClass('glyphicon-menu-down').addClass('glyphicon-menu-right');
          $(this).next().hide();
        }
      } else {
        _this.openEditorView(_path, function (res) {
          if (res.status) _tath.addClass('active');
        });
      }
      e.stopPropagation();
      e.preventDefault();
    });

    // 禁用目录选择（文件目录）
    $('.ace_catalogue').bind("selectstart", function (e) {
      var omitformtags = ["input", "textarea"];
      omitformtags = "|" + omitformtags.join("|") + "|";
      if (omitformtags.indexOf("|" + e.target.tagName.toLowerCase() + "|") == -1) {
        return false;
      } else {
        return true;
      }
    });
    // 返回目录（文件目录主菜单）
    $('.ace_dir_tools').on('click', '.upper_level', function () {
      var _paths = $(this).attr('data-menu-path');
      _this.reader_file_dir_menu({ path: _paths, is_empty: true });
      $('.ace_catalogue_title').html('目录：' + _paths).attr('title', _paths);
    });
    // 新建文件（文件目录主菜单）
    $('.ace_dir_tools').on('click', '.new_folder', function (e) {
      var _paths = $(this).parent().find('.upper_level').attr('data-menu-path');
      $(this).find('.folder_down_up').show();
      $(document).click(function () {
        $('.folder_down_up').hide();
        $(this).unbind('click');
        return false;
      });
      $('.ace_toolbar_menu').hide();
      $('.ace_catalogue_menu').hide();
      $('.ace_toolbar_menu .menu-tabs,.ace_toolbar_menu .menu-encoding,.ace_toolbar_menu .menu-files').hide();
      e.stopPropagation();
      e.preventDefault();
    });
    // 刷新列表 (文件目录主菜单)
    $('.ace_dir_tools').on('click', '.refresh_dir', function (e) {
      _this.refresh_config = {
        el: $('.cd-accordion-menu')[0],
        path: $('.ace_catalogue_title').attr('title'),
        group: 1,
        is_empty: true
      }
      _this.reader_file_dir_menu(_this.refresh_config, function () {
        layer.msg('刷新成功', { icon: 1 });
      });
    });
    // 搜索内容 (文件目录主菜单)
    $('.ace_dir_tools').on('click', '.search_file', function (e) {
      if ($(this).parent().find('.search_input_view').length == 0) {
        $(this).siblings('div').hide();
        $(this).css('color', '#ec4545').attr({ 'title': '关闭' }).find('.glyphicon').removeClass('glyphicon-search').addClass('glyphicon-remove').next().text("关闭");
        $(this).before('<div class="search_input_title">搜索目录文件</div>');
        $(this).after('<div class="search_input_view">\
					<form>\
                        <input type="text" id="search_input_val" class="ser-text pull-left" placeholder="">\
                        <button type="button" class="ser-sub pull-left"></button>\
                    </form>\
                    <div class="search_boxs">\
                        <input id="search_alls" type="checkbox">\
                        <label for="search_alls"><span>包含子目录文件</span></label>\
                    </div>\
                </div>');
        $('.ace_catalogue_list').css('top', '150px');
        $('.ace_dir_tools').css('height', '110px');
        $('.cd-accordion-menu').empty();
      } else {
        $(this).siblings('div').show();
        $(this).parent().find('.search_input_view,.search_input_title').remove();
        $(this).removeAttr('style').attr({ 'title': '搜索内容' }).find('.glyphicon').removeClass('glyphicon-remove').addClass('glyphicon-search').next().text("搜索");
        $('.ace_catalogue_list').removeAttr('style');
        $('.ace_dir_tools').removeAttr('style');
        _this.refresh_config = {
          el: $('.cd-accordion-menu')[0],
          path: $('.ace_catalogue_title').attr('title'),
          group: 1,
          is_empty: true
        }
        _this.reader_file_dir_menu(_this.refresh_config);
      }
    });

    // 搜索文件内容
    $('.ace_dir_tools').on('click', '.search_input_view button', function (e) {
      var path = _this.menu_path,
        search = $('#search_input_val').val();
      _this.reader_file_dir_menu({
        el: $('.cd-accordion-menu')[0],
        path: path,
        group: 1,
        search: search,
        all: $('#search_alls').is(':checked') ? 'True' : 'False',
        is_empty: true
      })
    });

    // 当前根目录操作，新建文件或目录
    $('.ace_dir_tools').on('click', '.folder_down_up li', function (e) {
      var _type = parseInt($(this).attr('data-type')), element = $('.cd-accordion-menu'), group = 0, type = 'Dir';
      if ($('.file_fold.bg').length > 0 && $('.file_fold.bg').data('file') !== 'files') {
        element = $('.file_fold.bg');
        group = parseInt(element.data('group'));
        type = element.data('file');
        if (type === 'Files' && group !== 0) {
          if (group === 1) {
            element = element.parent().parent()
          } else {
            element = element.parent().parent().prev()
          }
          group = group - 1;
        }
      }
      // console.log(element)
      switch (_type) {
        case 2:
          _this.newly_file_type_dom(element, group, 0);
          break;
        case 3:
          _this.newly_file_type_dom(element, group, 1);
          break;
      }
      _this.refresh_config = {
        el: $('.cd-accordion-menu')[0],
        path: $('.ace_catalogue_title').attr('title'),
        group: 1,
        is_empty: true
      }
      $(this).parent().hide();
      $('.ace_toolbar_menu').hide();
      $('.ace_toolbar_menu .menu-tabs,.ace_toolbar_menu .menu-encoding,.ace_toolbar_menu .menu-files').hide();
      e.preventDefault();
      e.stopPropagation();
    });
    // 移动编辑器文件目录
    $('.ace_catalogue_drag_icon .drag_icon_conter').on('mousedown', function (e) {
      var _left = $('.aceEditors')[0].offsetLeft;
      $('.ace_gutter-layer').css('cursor', 'col-resize');
      $('#ace_conter').unbind().on('mousemove', function (ev) {
        var _width = (ev.clientX + 1) - _left;
        if (_width >= 265 && _width <= 450) {
          $('.ace_catalogue').css({ 'width': _width, 'transition': 'none' });
          $('.ace_editor_main').css({ 'marginLeft': _width, 'transition': 'none' });
          $('.ace_catalogue_drag_icon ').css('left', _width);
          $('.file_fold .newly_file_input').width($('.file_fold .newly_file_input').parent().parent().parent().width() - ($('.file_fold .newly_file_input').parent().parent().attr('data-group') * 15 - 5) - 20 - 30 - 53);
        }
      }).on('mouseup', function (ev) {
        $('.ace_gutter-layer').css('cursor', 'inherit');
        $('.ace_catalogue').css('transition', 'all 500ms');
        $('.ace_editor_main').css('transition', 'all 500ms');
        $(this).unbind('mouseup mousemove');
      });
    });
    // 收藏目录显示和隐藏
    $('.ace_catalogue_drag_icon .fold_icon_conter').on('click', function (e) {
      if ($('.ace_overall').hasClass('active')) {
        $('.ace_overall').removeClass('active');
        $('.ace_catalogue').css('left', '0');
        $(this).removeClass('active').attr('title', '隐藏文件目录');
        $('.ace_editor_main').css('marginLeft', $('.ace_catalogue').width());
      } else {
        $('.ace_overall').addClass('active');
        $('.ace_catalogue').css('left', '-' + $('.ace_catalogue').width() + 'px');
        $(this).addClass('active').attr('title', '显示文件目录');
        $('.ace_editor_main').css('marginLeft', 0);
      }
      setTimeout(function () {
        if (_this.ace_active != '') _this.editor[_this.ace_active].ace.resize();
      }, 600);
    });
    // 恢复历史文件
    $('.ace_conter_tips').on('click', 'a', function () {
      _this.event_ecovery_file(this);
    });
    // 右键菜单
    $('.ace_catalogue_list').on('mousedown', '.has-children .file_fold', function (e) {
      var x = e.clientX, y = e.clientY, _left = $('.aceEditors')[0].offsetLeft, _top = $('.aceEditors')[0].offsetTop, _that = $('.ace_catalogue_list .has-children .file_fold'), _active = $('.ace_catalogue_list .has-children .file_fold.edit_file_group');
      $('.ace_toolbar_menu').hide();
      if (e.which === 3) {
        if ($(this).hasClass('edit_file_group')) return false;
        $('.ace_catalogue_menu').css({ 'display': 'block', 'left': x + 15, 'top': y + 10 });
        _that.removeClass('bg');
        $(this).addClass('bg');
        _active.attr('data-edit') != '2' ? _active.parent().remove() : '';
        _that.removeClass('edit_file_group').removeAttr('data-edit');
        _that.find('.file_input').siblings().show();
        _that.find('.file_input').remove();
        $('.ace_catalogue_menu li').show();
        if ($(this).attr('data-file') == 'Dir') {
          $('.ace_catalogue_menu li:nth-child(6)').hide();
        } else {
          $('.ace_catalogue_menu li:nth-child(-n+4)').hide();
        }
        $(document).click(function () {
          $('.ace_catalogue_menu').hide();
          $(this).unbind('click');
          return false;
        });
        _this.refresh_config = {
          el: $(this).parent().parent()[0],
          path: _this.get_file_dir($(this).parent().attr('data-menu-path'), 1),
          group: parseInt($(this).attr('data-group')),
          is_empty: true
        }
      }
    });
    // 文件目录右键功能
    $('.ace_catalogue_menu li').click(function (e) {
      _this.newly_file_type(this);
    });
    // 新建、重命名鼠标事件
    $('.ace_catalogue_list').on('click', '.has-children .edit_file_group .glyphicon-ok', function () {
      var _file_or_dir = $(this).parent().find('input').val(),
        _file_type = $(this).parent().parent().attr('data-file'),
        _path = $('.has-children .file_fold.bg').parent().attr('data-menu-path'),
        _type = parseInt($(this).parent().parent().attr('data-edit'));
      if ($(this).parent().parent().parent().attr('data-menu-path') === undefined && parseInt($(this).parent().parent().attr('data-group')) === 1) {
        // console.log('根目录')
        _path = $('.ace_catalogue_title').attr('title');
      }
      // 			return false;
      if (_file_or_dir === '') {
        $(this).prev().css('border', '1px solid #f34a4a');
        layer.tips(_type === 0 ? '文件目录不能为空' : (_type === 1 ? '文件名称不能空' : '新名称不能为空'), $(this).prev(), { tips: [1, '#f34a4a'], time: 0 });
        return false;
      } else if ($(this).prev().attr('data-type') == 0) {
        return false;
      }
      switch (_type) {
        case 0: //新建文件夹
          _this.event_create_dir({ path: _path + '/' + _file_or_dir });
          break;
        case 1: //新建文件
          _this.event_create_file({ path: _path + '/' + _file_or_dir });
          break;
        case 2: //重命名
          _this.event_rename_currency({ sfile: _path, dfile: _this.get_file_dir(_path, 1) + '/' + _file_or_dir });
          break;
      }
    });
    // 新建、重命名键盘事件
    $('.ace_catalogue_list').on('keyup', '.has-children .edit_file_group input', function (e) {
      var _type = $(this).parent().parent().attr('data-edit'),
        _arry = $('.has-children .file_fold.bg+ul>li');
      if (_arry.length == 0 && $(this).parent().parent().attr('data-group') == 1) _arry = $('.cd-accordion-menu>li')
      if (_type != 2) {
        for (var i = 0; i < _arry.length; i++) {
          if ($(_arry[i]).find('.file_title span').html() === $(this).val()) {
            $(this).css('border', '1px solid #f34a4a');
            $(this).attr('data-type', 0);
            layer.tips(_type == 0 ? '文件目录存在同名目录' : '文件名称存在同名文件', $(this)[0], { tips: [1, '#f34a4a'], time: 0 });
            return false
          }
        }
      }
      if (_type == 1 && $(this).val().indexOf('.')) $(this).prev().removeAttr('class').addClass(_this.get_file_suffix($(this).val()) + '-icon');
      $(this).attr('data-type', 1);
      $(this).css('border', '1px solid #528bff');
      layer.closeAll('tips');
      if (e.keyCode === 13) $(this).next().click();
      $('.ace_toolbar_menu').hide();
      $('.ace_toolbar_menu .menu-tabs,.ace_toolbar_menu .menu-encoding,.ace_toolbar_menu .menu-files').hide();
      e.stopPropagation();
      e.preventDefault();
    });
    // 新建、重命名鼠标点击取消事件
    $('.ace_catalogue_list').on('click', '.has-children .edit_file_group .glyphicon-remove', function () {
      layer.closeAll('tips');
      if ($(this).parent().parent().parent().attr('data-menu-path')) {
        $(this).parent().parent().removeClass('edit_file_group').removeAttr('data-edit');
        $(this).parent().siblings().show();
        $(this).parent().remove();
        return false;
      }
      $(this).parent().parent().parent().remove();
    });
    //屏蔽浏览器右键菜单
    $('.ace_catalogue_list')[0].oncontextmenu = function () {
      return false;
    }
    $('.ace_conter_menu').dragsort({
      dragSelector: '.icon_file',
      itemSelector: 'li'
    });
    this.setEditorView();
    this.reader_file_dir_menu();
  },
  // 	设置本地存储，设置类型type：session或local
  setStorage: function (type, key, val) {
    if (type != "local" && type != "session") val = key, key = type, type = 'session';
    window[type + 'Storage'].setItem(key, val);
  },
  //获取指定本地存储，设置类型type：session或local
  getStorage: function (type, key) {
    if (type != "local" && type != "session") key = type, type = 'session';
    return window[type + 'Storage'].getItem(key);
  },
  //删除指定本地存储，设置类型type：session或local
  removeStorage: function (type, key) {
    if (type != "local" && type != "session") key = type, type = 'session';
    window[type + 'Storage'].removeItem(key);
  },
  // 	删除指定类型的所有存储信息
  clearStorage: function (type) {
    if (type != "local" && type != "session") key = type, type = 'session';
    window[type + 'Storage'].clear();
  },

  // 新建文件类型
  newly_file_type: function (that) {
    var _type = parseInt($(that).attr('data-type')),
      _active = $('.ace_catalogue .ace_catalogue_list .has-children .file_fold.bg'),
      _group = parseInt(_active.attr('data-group')),
      _path = _active.parent().attr('data-menu-path'), //当前文件夹新建
      _this = this;
    // console.log(_type);
    switch (_type) {
      case 0: //刷新目录
        _active.next().empty();
        _this.reader_file_dir_menu({
          el: _active.next(),
          path: _path,
          group: parseInt(_active.attr('data-group')) + 1,
          is_empty: true
        }, function () {
          layer.msg('刷新成功', { icon: 1 });
        });
        break;
      case 1: //打开文件
        _this.menu_path = _path;
        _this.reader_file_dir_menu({
          el: '.cd-accordion-menu',
          path: _this.menu_path,
          group: 1,
          is_empty: true
        });
        break;
      case 2: //新建文件
      case 3:
        if (this.get_file_dir(_path, 1) != this.menu_path) { //判断当前文件上级是否为显示根目录
          this.reader_file_dir_menu({ el: _active, path: _path, group: _group + 1 }, function (res) {
            _this.newly_file_type_dom(_active, _group, _type == 2 ? 0 : 1);
          });
        } else {
          _this.newly_file_type_dom(_active, _group, _type == 2 ? 0 : 1);
        }
        break;
      case 4: //文件重命名
        var _types = _active.attr('data-file');
        if (_active.hasClass('active')) {
          layer.msg('该文件已打开，无法修改名称', { icon: 0 });
          return false;
        }
        _active.attr('data-edit', 2);
        _active.addClass('edit_file_group');
        _active.find('.file_title').hide();
        _active.find('.glyphicon').hide();
        _active.prepend('<span class="file_input"><i class="' + (_types === 'Dir' ? 'folder' : (_this.get_file_suffix(_active.find('.file_title span').html()))) + '-icon"></i><input type="text" class="newly_file_input" value="' + (_active.find('.file_title span').html()) + '"><span class="glyphicon glyphicon-ok" aria-hidden="true"></span><span class="glyphicon glyphicon-remove" aria-hidden="true"></span>')
        $('.file_fold .newly_file_input').width($('.file_fold .newly_file_input').parent().parent().parent().width() - ($('.file_fold .newly_file_input').parent().parent().attr('data-group') * 15 - 5) - 20 - 30 - 53);
        $('.file_fold .newly_file_input').focus();
        break;
      case 5:
        window.open('/download?filename=' + encodeURIComponent(_path));
        break;
      case 6:
        var is_files = _active.attr('data-file') === 'Files'
        layer.confirm(lan.get(is_files ? 'recycle_bin_confirm' : 'recycle_bin_confirm_dir', [_active.find('.file_title span').html()]), { title: is_files ? lan.files.del_file : lan.files.del_dir, closeBtn: 2, icon: 3 }, function (index) {
          _this[is_files ? 'del_file_req' : 'del_dir_req']({ path: _path }, function (res) {
            layer.msg(res.msg, { icon: res.status ? 1 : 2 });
            if (res.status) {
              if (_active.attr('data-group') != 1) _active.parent().parent().prev().addClass('bg')
              _this.reader_file_dir_menu(_this.refresh_config, function () {
                layer.msg(res.msg, { icon: 1 });
              });
            }
          });
        });
        break;
    }
  },
  // 新建文件和文件夹
  newly_file_type_dom: function (_active, _group, _type, _val) {
    var _html = '', _this = this, _nextLength = _active.next(':not(.ace_catalogue_menu)').length;
    if (_nextLength > 0) {
      _active.next().show();
      _active.find('.glyphicon').removeClass('glyphicon-menu-right').addClass('glyphicon-menu-down');
    }
    _html += '<li class="has-children children_' + (_group + 1) + '"><div class="file_fold edit_file_group group_' + (_group + 1) + '" data-group="' + (_group + 1) + '" data-edit="' + _type + '"><span class="file_input">';
    _html += '<i class="' + (_type == 0 ? 'folder' : (_type == 1 ? 'text' : (_this.get_file_suffix(_val || '')))) + '-icon"></i>'
    _html += '<input type="text" class="newly_file_input" value="' + (_val != undefined ? _val : '') + '">'
    _html += '<span class="glyphicon glyphicon-ok" aria-hidden="true"></span><span class="glyphicon glyphicon-remove" aria-hidden="true"></span></span></div></li>'
    if (_nextLength > 0) {
      _active.next().prepend(_html);
    } else {
      _active.prepend(_html);
    }
    setTimeout(function () {
      $('.newly_file_input').focus()
    }, 100)
    $('.file_fold .newly_file_input').width($('.file_fold .newly_file_input').parent().parent().parent().width() - ($('.file_fold .newly_file_input').parent().parent().attr('data-group') * 15 - 5) - 20 - 30 - 53);
    return false;
  },
  // 通用重命名事件
  event_rename_currency: function (obj) {
    var _active = $('.ace_catalogue_list .has-children .file_fold.edit_file_group'), _this = this;
    this.rename_currency_req({ sfile: obj.sfile, dfile: obj.dfile }, function (res) {
      layer.msg(res.msg, { icon: res.status ? 1 : 2 });
      if (res.status) {
        _this.reader_file_dir_menu(_this.refresh_config, function () {
          layer.msg(res.msg, { icon: 1 });
        });
      } else {
        _active.find('.file_input').siblings().show();
        _active.find('.file_input').remove();
        _active.removeClass('edit_file_group').removeAttr('data-edit');
      }
    })
  },
  // 创建文件目录事件
  event_create_dir: function (obj) {
    var _this = this;
    this.create_dir_req({ path: obj.path }, function (res) {
      layer.msg(res.msg, { icon: res.status ? 1 : 2 });
      if (res.status) {
        _this.reader_file_dir_menu(_this.refresh_config, function () {
          layer.msg(res.msg, { icon: 1 });
        });
      }
    })
  },
  // 创建文件事件
  event_create_file: function (obj) {
    var _this = this;
    this.create_file_req({ path: obj.path }, function (res) {
      layer.msg(res.msg, { icon: res.status ? 1 : 2 });
      if (res.status) {
        _this.reader_file_dir_menu(_this.refresh_config, function () {
          layer.msg(res.msg, { icon: 1 });
          _this.openEditorView(obj.path);
        });
      }
    })
  },
  // 重命名请求
  rename_currency_req: function (obj, callback) {
    var loadT = layer.msg('正在重命名文件或目录，请稍候...', { time: 0, icon: 16, shade: [0.3, '#000'] });
    $.post("/files?action=MvFile", {
      sfile: obj.sfile,
      dfile: obj.dfile,
      rename: 'true'
    }, function (res) {
      layer.close(loadT);
      if (callback) callback(res);
    });
  },
  // 创建文件事件
  create_file_req: function (obj, callback) {
    var loadT = layer.msg('正在新建文件，请稍候...', { time: 0, icon: 16, shade: [0.3, '#000'] });
    $.post("/files?action=CreateFile", {
      path: obj.path
    }, function (res) {
      layer.close(loadT);
      if (callback) callback(res);
    });
  },
  // 创建目录请求
  create_dir_req: function (obj, callback) {
    var loadT = layer.msg('正在新建目录，请稍候...', { time: 0, icon: 16, shade: [0.3, '#000'] });
    $.post("/files?action=CreateDir", {
      path: obj.path
    }, function (res) {
      layer.close(loadT);
      if (callback) callback(res);
    });
  },
  // 删除文件请求
  del_file_req: function (obj, callback) {
    var loadT = layer.msg('正在删除文件，请稍候...', { time: 0, icon: 16, shade: [0.3, '#000'] });
    $.post("/files?action=DeleteFile", {
      path: obj.path
    }, function (res) {
      layer.close(loadT);
      if (callback) callback(res);
    });
  },
  // 删除目录请求
  del_dir_req: function (obj, callback) {
    var loadT = layer.msg('正在删除文件目录，请稍候...', { time: 0, icon: 16, shade: [0.3, '#000'] });
    $.post("/files?action=DeleteDir", {
      path: obj.path
    }, function (res) {
      layer.close(loadT);
      if (callback) callback(res);
    });
  },
  // 临时文件保存
  auto_save_temp: function (obj, callback) {
    // var loadT = layer.msg('正在新建目录，请稍候...',{time: 0,icon: 16,shade: [0.3, '#000']});
    $.post("/files?action=auto_save_temp", {
      filename: obj.filename,
      body: obj.body
    }, function (res) {
      layer.close(loadT);
      if (callback) callback(res);
    });
  },
  // 获取临时文件内容
  get_auto_save_body: function (obj, callback) {
    var loadT = layer.msg('正在获取自动保存文件信息，请稍候...', { time: 0, icon: 16, shade: [0.3, '#000'] });
    $.post("/files?action=get_auto_save_body", {
      filename: obj.filename
    }, function (res) {
      layer.close(loadT);
      if (callback) callback(res);
    });
  },
  // 恢复历史文件事件
  event_ecovery_file: function (that) {
    var _path = $(that).attr('data-path'), _history = new Number($(that).attr('data-history')), _this = this;
    var loadT = layer.open({
      type: 1,
      area: ['400px', '180px'],
      title: '恢复历史文件',
      content: '<div class="ace-clear-form">\
				<div class="clear-icon"></div>\
				<div class="clear-title">是否恢复历史文件&nbsp<span class="size_ellipsis" style="max-width:150px;vertical-align: top;" title="' + bt.format_data(_history) + '">' + bt.format_data(_history) + '</span>?</div>\
				<div class="clear-tips">恢复历史文件后，当前文件内容将会被替换！</div>\
				<div class="ace-clear-btn" style="">\
					<button type="button" class="btn btn-sm btn-default" style="margin-right:10px;" data-type="1">取消</button>\
					<button type="button" class="btn btn-sm btn-success" data-type="0">恢复历史文件</button>\
				</div>\
			</div>',
      success: function (layero, index) {
        $('.ace-clear-btn .btn').click(function () {
          var _type = $(this).attr('data-type');
          switch (_type) {
            case '0':
              _this.recovery_file_history({
                filename: _path,
                history: _history
              }, function (res) {
                layer.close(index);
                layer.msg(res.status ? '恢复历史文件成功' : '恢复历史文件失败', { icon: res.status ? 1 : 2 });
                if (res.status) {
                  if (_this.editor[_this.ace_active].historys_file) {
                    _this.removeEditor(_this.ace_active);
                  }
                  if ($('.ace_conter_menu>[title="' + _path + '"]').length > 0) {
                    $('.ace_header .refreshs').click();
                    layer.close(_this.layer_view);
                  }
                }
              });
              break;
            case '1':
              layer.close(index);
              break;
          }
        });
      }
    });
  },
  // 判断是否为历史文件
  is_file_history: function (_item) {
    if (_item == undefined) return false;
    if (_item.historys_file) {
      $('.ace_conter_tips').show();
      $('#ace_editor_' + _item.id).css('bottom', '50px');
      $('.ace_conter_tips .tips').html('只读文件，文件为' + _item.path + '，历史版本 [ ' + bt.format_data(new Number(_item.historys_active)) + ' ]<a href="javascript:;" class="ml35 btlink" style="margin-left:35px" data-path="' + _item.path + '" data-history="' + _item.historys_active + '">点击恢复当前历史版本</a>');
    } else {
      $('.ace_conter_tips').hide();
    }
  },
  // 判断文件是否打开
  is_file_open: function (path, callabck) {
    var is_state = false
    for (var i = 0; i < this.pathAarry.length; i++) {
      if (path === this.pathAarry[i]) is_state = true
    }
    if (callabck) {
      callabck(is_state);
    } else {
      return is_state;
    }
  },
  // 恢复文件历史
  recovery_file_history: function (obj, callback) {
    var loadT = layer.msg('正在恢复历史文件，请稍候...', { time: 0, icon: 16, shade: [0.3, '#000'] });
    $.post("/files?action=re_history", {
      filename: obj.filename,
      history: obj.history
    }, function (res) {
      layer.close(loadT);
      if (callback) callback(res);
    });
  },
  // 获取文件列表
  get_file_dir_list: function (obj, callback) {
    var loadT = layer.msg('正在获取文件列表，请稍候...', { time: 0, icon: 16, shade: [0.3, '#000'] }), _this = this;
    if (obj['p'] === undefined) obj['p'] = 1;
    if (obj['showRow'] === undefined) obj['showRow'] = 200;
    if (obj['sort'] === undefined) obj['sort'] = 'name';
    if (obj['reverse'] === undefined) obj['reverse'] = 'False';
    if (obj['search'] === undefined) obj['search'] = '';
    if (obj['all'] === undefined) obj['all'] = 'False';
    $.post("/files?action=GetDir&tojs=GetFiles", { p: obj.p, showRow: obj.showRow, sort: obj.sort, reverse: obj.reverse, path: obj.path, search: obj.search }, function (res) {
      layer.close(loadT);
      if (callback) callback(res);
    });
  },
  // 获取历史文件
  get_file_history: function (obj, callback) {
    var loadT = layer.msg('正在获取历史文件内容，请稍候...', { time: 0, icon: 16, shade: [0.3, '#000'] }), _this = this;
    $.post("/files?action=read_history", { filename: obj.filename, history: obj.history }, function (res) {
      layer.close(loadT);
      if (callback) callback(res);
    });
  },
  // 渲染文件列表
  reader_file_dir_menu: function (obj, callback) {
    var _path = getCookie('Path'), _this = this;
    if (obj === undefined) obj = {}
    if (obj['el'] === undefined) obj['el'] = '.cd-accordion-menu';
    if (obj['group'] === undefined) obj['group'] = 1;
    if (obj['p'] === undefined) obj['p'] = 1;
    if (obj['path'] === undefined) obj['path'] = _path;
    if (obj['search'] === undefined) obj['search'] = '';
    if (obj['is_empty'] === undefined) obj['is_empty'] = false;
    if (obj['all'] === undefined) obj['all'] = 'False'
    this.get_file_dir_list({ p: obj.p, path: obj.path, search: obj.search, all: obj.all }, function (res) {
      var _dir = res.DIR, _files = res.FILES, _dir_dom = '', _files_dom = '', _html = '';
      _this.menu_path = res.PATH;
      for (var i = 0; i < _dir.length; i++) {
        var _data = _dir[i].split(';');
        if (_data[0] === '__pycache__') continue;
        _dir_dom += '<li class="has-children children_' + obj.group + '" title="' + (obj.path + '/' + _data[0]) + '" data-menu-path="' + (obj.path + '/' + _data[0]) + '" data-size="' + (_data[1]) + '">\
					<div class="file_fold group_'+ obj.group + '" data-group="' + obj.group + '" data-file="Dir">\
						<span class="glyphicon glyphicon-menu-right"></span>\
						<span class="file_title"><i class="folder-icon"></i><span>'+ _data[0] + '</span></span>\
					</div>\
					<ul data-group=""></ul>\
					<span class="has_children_separator"></span>\
				</li>';
      }
      for (var j = 0; j < _files.length; j++) {
        var _data = _files[j].split(';');
        if (_data[0].indexOf('.pyc') !== -1) continue;
        _files_dom += '<li class="has-children" title="' + (obj.path + '/' + _data[0]) + '" data-menu-path="' + (obj.path + '/' + _data[0]) + '" data-size="' + (_data[1]) + '" data-suffix="' + _this.get_file_suffix(_data[0]) + '">\
					<div class="file_fold  group_'+ obj.group + '" data-group="' + obj.group + '" data-file="Files">\
						<span class="file_title"><i class="'+ _this.get_file_suffix(_data[0]) + '-icon"></i><span>' + _data[0] + '</span></span>\
					</div>\
				</li>';
      }
      if (res.PATH !== '/' && obj['group'] === 1) {
        $('.upper_level').attr('data-menu-path', _this.get_file_dir(res.PATH, 1));
        $('.ace_catalogue_title').html('目录：' + res.PATH).attr('title', res.PATH);
        $('.upper_level').html('<i class="glyphicon glyphicon-share-alt" aria-hidden="true"></i>上一级')
      } else if (res.PATH === '/') {
        $('.upper_level').html('<i class="glyphicon glyphicon-hdd" aria-hidden="true"></i>根目录')
      }
      if (obj.is_empty) $(obj.el).empty();
      $(obj.el).append(_html + _dir_dom + _files_dom);
      if (callback) callback(res);
    });
  },
  // 获取文件目录位置
  get_file_dir: function (path, num) {
    var _arry = path.split('/');
    if (path === '/') return '/';
    _arry.splice(-1, num);
    return _arry == '' ? '/' : _arry.join('/');
  },
  // 获取文件全称
  get_file_suffix: function (fileName) {
    var filenames = fileName.match(/\.([0-9A-z]*)$/);
    filenames = (filenames == null ? 'text' : filenames[1]);
    for (var name in this.aceConfig.supportedModes) {
      var data = this.aceConfig.supportedModes[name], suffixs = data[0].split('|'), filename = name.toLowerCase();
      for (var i = 0; i < suffixs.length; i++) {
        if (filenames == suffixs[i]) return filename;
      }
    }
    return 'text';
  },
  // 设置编辑器视图
  setEditorView: function () {
    var aceEditorHeight = $('.aceEditors').height(), _this = this;
    var autoAceHeight = setInterval(function () {
      var page_height = $('.aceEditors').height();
      var ace_conter_menu = $('.ace_conter_menu').height();
      var ace_conter_toolbar = $('.ace_conter_toolbar').height();
      var _height = page_height - ($('.pull-down .glyphicon').hasClass('glyphicon-menu-down') ? 35 : 0) - ace_conter_menu - ace_conter_toolbar - 42;
      $('.ace_conter_editor').height(_height);
      if (aceEditorHeight == $('.aceEditors').height()) {
        if (_this.ace_active) _this.editor[_this.ace_active].ace.resize();
        clearInterval(autoAceHeight);
      } else {
        aceEditorHeight = $('.aceEditors').height();
      }
    }, 200);
  },
  // 获取文件编码列表
  getEncodingList: function (type) {
    var _option = '';
    for (var i = 0; i < this.aceConfig.encodingList.length; i++) {
      var item = this.aceConfig.encodingList[i] == type.toUpperCase();
      _option += '<li data- data-value="' + this.aceConfig.encodingList[i] + '" ' + (item ? 'class="active"' : '') + '>' + this.aceConfig.encodingList[i] + (item ? '<span class="icon"><i class="glyphicon glyphicon-ok" aria-hidden="true"></i></span>' : '') + '</li>';
    }
    $('.menu-encoding ul').html(_option);
  },
  // 获取文件关联列表
  getRelevanceList: function (fileName) {
    var _option = '', _top = 0, fileType = this.getFileType(fileName), _set_tops = 0;
    for (var name in this.aceConfig.supportedModes) {
      var data = this.aceConfig.supportedModes[name], item = (name == fileType.name);
      _option += '<li data-height="' + _top + '" data-rule="' + this.aceConfig.supportedModes[name] + '" data-value="' + name + '" ' + (item ? 'class="active"' : '') + '>' + (this.aceConfig.nameOverrides[name] || name) + (item ? '<span class="icon"><i class="glyphicon glyphicon-ok" aria-hidden="true"></i></span>' : '') + '</li>'
      if (item) _set_tops = _top
      _top += 35;
    }
    $('.menu-files ul').html(_option);
    $('.menu-files ul').scrollTop(_set_tops);
  },
  // 搜索文件关联
  searchRelevance: function (search) {
    if (search == undefined) search = '';
    $('.menu-files ul li').each(function (index, el) {
      var val = $(this).attr('data-value').toLowerCase(),
        rule = $(this).attr('data-rule'),
        suffixs = rule.split('|'),
        _suffixs = false;
      search = search.toLowerCase();
      for (var i = 0; i < suffixs.length; i++) {
        if (suffixs[i].indexOf(search) > -1) _suffixs = true
      }
      if (search == '') {
        $(this).removeAttr('style');
      } else {
        if (val.indexOf(search) == -1) {
          $(this).attr('style', 'display:none');
        } else {
          $(this).removeAttr('style');
        }
        if (_suffixs) $(this).removeAttr('style')
      }
    });
  },
  // 设置编码类型
  setEncodingType: function (encode) {
    this.getEncodingList('UTF-8');
    $('.menu-encoding ul li').click(function (e) {
      layer.msg('设置文件编码：' + $(this).attr('data-value'));
      $(this).addClass('active').append('<span class="icon"><i class="glyphicon glyphicon-ok" aria-hidden="true"></i></span>').siblings().removeClass('active').find('span').remove();
    });
  },
  // 更新状态栏
  currentStatusBar: function (id) {
    var _item = this.editor[id];
    if (_item == undefined) {
      this.removerStatusBar();
      return false;
    }
    $('.ace_conter_toolbar [data-type="cursor"]').html('行<i class="cursor-row">1</i>,列<i class="cursor-line">0</i>');
    $('.ace_conter_toolbar [data-type="history"]').html('历史版本：<i>' + (_item.historys.length === 0 ? '无' : _item.historys.length + '份') + '</i>');
    $('.ace_conter_toolbar [data-type="path"]').html('文件位置：<i title="' + _item.path + '">' + _item.path + '</i>');
    $('.ace_conter_toolbar [data-type="tab"]').html(_item.softTabs ? '空格：<i>' + _item.tabSize + '</i>' : '制表符长度：<i>' + _item.tabSize + '</i>');
    $('.ace_conter_toolbar [data-type="encoding"]').html('编码：<i>' + _item.encoding.toUpperCase() + '</i>');
    $('.ace_conter_toolbar [data-type="lang"]').html('语言：<i>' + _item.type + '</i>');
    $('.ace_conter_toolbar span').attr('data-id', id);
    $('.file_fold').removeClass('bg');
    $('[data-menu-path="' + (_item.path) + '"]').find('.file_fold').addClass('bg');
    if (_item.historys_file) {
      $('.ace_conter_toolbar [data-type="history"]').hide();
    } else {
      $('.ace_conter_toolbar [data-type="history"]').show();
    }
    _item.ace.resize();
  },
  // 清除状态栏
  removerStatusBar: function () {
    $('.ace_conter_toolbar [data-type="history"]').html('');
    $('.ace_conter_toolbar [data-type="path"]').html('');
    $('.ace_conter_toolbar [data-type="tab"]').html('');
    $('.ace_conter_toolbar [data-type="cursor"]').html('');
    $('.ace_conter_toolbar [data-type="encoding"]').html('');
    $('.ace_conter_toolbar [data-type="lang"]').html('');
  },
  // 创建ACE编辑器-对象
  creationEditor: function (obj, callabck) {
    var _this = this;
    $('#ace_editor_' + obj.id).text(obj.data || '');
    $('.ace_conter_editor .ace_editors').css('fontSize', _this.fontSize + 'px');
    if (this.editor == null) this.editor = {};
    this.editor[obj.id] = {
      ace: ace.edit("ace_editor_" + obj.id, {
        theme: "ace/theme/" + _this.aceConfig.aceEditor.editorTheme, //主题
        mode: "ace/mode/" + (obj.fileName != undefined ? obj.mode : 'text'), // 语言类型
        wrap: _this.aceConfig.aceEditor.wrap,
        showInvisibles: _this.aceConfig.aceEditor.showInvisibles,
        showPrintMargin: false,
        enableBasicAutocompletion: true,
        enableSnippets: _this.aceConfig.aceEditor.enableSnippets,
        enableLiveAutocompletion: _this.aceConfig.aceEditor.enableLiveAutocompletion,
        useSoftTabs: _this.aceConfig.aceEditor.useSoftTabs,
        tabSize: _this.aceConfig.aceEditor.tabSize,
        keyboardHandler: 'sublime',
        readOnly: obj.readOnly === undefined ? false : obj.readOnly
      }), //ACE编辑器对象
      id: obj.id,
      wrap: _this.aceConfig.aceEditor.wrap, //是否换行
      path: obj.path,
      tabSize: _this.aceConfig.aceEditor.tabSize,
      softTabs: _this.aceConfig.aceEditor.useSoftTabs,
      fileName: obj.fileName,
      enableSnippets: true, //是否代码提示
      encoding: (obj.encoding != undefined ? obj.encoding : 'utf-8'), //编码类型
      mode: (obj.fileName != undefined ? obj.mode : 'text'), //语言类型
      type: obj.type,
      fileType: 0, //文件状态 
      historys: obj.historys,
      historys_file: obj.historys_file === undefined ? false : obj.historys_file,
      historys_active: obj.historys_active === '' ? false : obj.historys_active
    };
    var ACE = this.editor[obj.id];
    ACE.ace.moveCursorTo(0, 0); //设置鼠标焦点
    ACE.ace.focus();//设置焦点
    ACE.ace.resize(); //设置自适应
    ACE.ace.commands.addCommand({
      name: '保存文件',
      bindKey: {
        win: 'Ctrl-S',
        mac: 'Command-S'
      },
      exec: function (editor) {
        _this.saveFileMethod(ACE);
      },
      readOnly: false // 如果不需要使用只读模式，这里设置false
    });
    ACE.ace.commands.addCommand({
      name: '跳转行',
      bindKey: {
        win: 'Ctrl-I',
        mac: 'Command-I'
      },
      exec: function (editor) {
        $('.ace_header .jumpLine').click();
      },
      readOnly: false // 如果不需要使用只读模式，这里设置false
    })
    // 获取光标位置
    ACE.ace.getSession().selection.on('changeCursor', function (e) {
      var _cursor = ACE.ace.selection.getCursor();
      $('[data-type="cursor"]').html('行<i class="cursor-row">' + (_cursor.row + 1) + '</i>,列<i class="cursor-line">' + _cursor.column + '</i>');
    });

    // 触发修改内容
    ACE.ace.getSession().on('change', function (editor) {
      $('.item_tab_' + ACE.id + ' .icon-tool').addClass('glyphicon-exclamation-sign').removeClass('glyphicon-remove').attr('data-file-state', '1');
      ACE.fileType = 1;
      $('.ace_toolbar_menu').hide();
    });
    this.currentStatusBar(ACE.id);
    this.is_file_history(ACE);
  },
  // 保存文件方法
  saveFileMethod: function (ACE) {
    if ($('.item_tab_' + ACE.id + ' .icon-tool').attr('data-file-state') == 0) {
      layer.msg('当前文件未修改，无需保存!');
      return false;
    }
    $('.item_tab_' + ACE.id + ' .icon-tool').attr('title', '保存文件中，请稍候..').removeClass('glyphicon-exclamation-sign').addClass('glyphicon-repeat');
    layer.msg('保存文件中，请稍候<img src="/static/img/ns-loading.gif" style="width:15px;margin-left:5px">', { icon: 0 });
    this.saveFileBody({
      path: ACE.path,
      data: ACE.ace.getValue(),
      encoding: ACE.encoding
    }, function (res) {
      ACE.fileType = 0;
      $('.item_tab_' + ACE.id + ' .icon-tool').attr('data-file-state', '0').removeClass('glyphicon-repeat').addClass('glyphicon-remove');
    }, function (res) {
      ACE.fileType = 1;
      $('.item_tab_' + ACE.id + ' .icon-tool').attr('data-file-state', '1').removeClass('glyphicon-remove').addClass('glyphicon-repeat');
    });
  },
  // 获取文件模型
  getFileType: function (fileName) {
    var filenames = fileName.match(/\.([0-9A-z]*)$/);
    filenames = (filenames == null ? 'text' : filenames[1]);
    for (var name in this.aceConfig.supportedModes) {
      var data = this.aceConfig.supportedModes[name], suffixs = data[0].split('|'), filename = name.toLowerCase();
      for (var i = 0; i < suffixs.length; i++) {
        if (filenames == suffixs[i]) {
          return { name: name, mode: filename };
        }
      }
    }
    return { name: 'Text', mode: 'text' };
  },
  // 新建编辑器视图-方法
  addEditorView: function (type, conifg) {
    if (type == undefined) type = 0
    var _index = this.editorLength, _id = bt.get_random(8);
    $('.ace_conter_menu .item').removeClass('active');
    $('.ace_conter_editor .ace_editors').removeClass('active');
    $('.ace_conter_menu').append('<li class="item active item_tab_' + _id + '" data-type="shortcutKeys" data-id="' + _id + '" >\
			<div class="ace_item_box">\
				<span class="icon_file"><i class="text-icon"></i></span>\
				<span>'+ (type ? conifg.title : ('新建文件-' + _index)) + '</span>\
				<i class="glyphicon icon-tool glyphicon-remove" aria-hidden="true" data-file-state="0" data-title="'+ (type ? conifg.title : ('新建文件-' + _index)) + '"></i>\
			</div>\
		</li>');
    $('#ace_editor_' + _id).siblings().removeClass('active');
    $('.ace_conter_editor').append('<div id="ace_editor_' + _id + '" class="ace_editors active">' + (type ? aceShortcutKeys.innerHTML : '') + '</div>');
    switch (type) {
      case 0:
        this.creationEditor({ id: _id });
        this.editorLength = this.editorLength + 1;
        break;
      case 1:
        this.removerStatusBar();
        this.editorLength = this.editorLength + 1;
        break;
    }
  },
  // 删除编辑器视图-方法
  removeEditor: function (id) {
    if (id == undefined) id = this.ace_active;
    if ($('.item_tab_' + id).next().length != 0 && this.editorLength != 1) {
      $('.item_tab_' + id).next().click();
    } else if ($('.item_tab_' + id).prev.length != 0 && this.editorLength != 1) {
      $('.item_tab_' + id).prev().click();
    }
    $('.item_tab_' + id).remove();
    $('#ace_editor_' + id).remove();
    this.editorLength--;
    if (this.editor[id] == undefined) return false;
    for (var i = 0; i < this.pathAarry.length; i++) {
      if (this.pathAarry[i] == this.editor[id].path) {
        this.pathAarry.splice(i, 1);
      }
    }
    if (!this.editor[id].historys_file) $('[data-menu-path="' + (this.editor[id].path) + '"]').find('.file_fold').removeClass('active bg');
    this.editor[id].ace.destroy();
    delete this.editor[id];
    if (this.editorLength === 0) {
      this.ace_active = '';
      this.pathAarry = [];
      this.removerStatusBar();
    } else {
      this.currentStatusBar(this.ace_active);
    }
    if (this.ace_active != '') this.is_file_history(this.editor[this.ace_active]);
  },
  // 打开历史文件文件-方法
  openHistoryEditorView: function (obj, callback) {
    // 文件类型（type，列如：JavaScript） 、文件模型（mode，列如：text）、文件标识（id,列如：x8AmsnYn）、文件编号（index,列如：0）、文件路径 (path，列如：/www/root/)
    var _this = this, path = obj.filename, paths = path.split('/'), _fileName = paths[paths.length - 1], _fileType = this.getFileType(_fileName), _type = _fileType.name, _mode = _fileType.mode, _id = bt.get_random(8), _index = this.editorLength;
    this.get_file_history({ filename: obj.filename, history: obj.history }, function (res) {
      _this.pathAarry.push(path);
      $('.ace_conter_menu .item').removeClass('active');
      $('.ace_conter_editor .ace_editors').removeClass('active');
      $('.ace_conter_menu').append('<li class="item active item_tab_' + _id + '" title="' + path + '" data-type="' + _type + '" data-mode="' + _mode + '" data-id="' + _id + '" data-fileName="' + _fileName + '">' +
        '<div class="ace_item_box">' +
        '<span class="icon_file"><img src="/static/img/ico-history.png"></span><span title="' + path + ' 历史版本[ ' + bt.format_data(obj.history) + ' ]' + '">' + _fileName + '</span>' +
        '<i class="glyphicon glyphicon-remove icon-tool" aria-hidden="true" data-file-state="0" data-title="' + _fileName + '"></i>' +
        '</div>' +
        '</li>');
      $('.ace_conter_editor').append('<div id="ace_editor_' + _id + '" class="ace_editors active"></div>');
      $('[data-paths="' + path + '"]').find('.file_fold').addClass('active bg');
      _this.ace_active = _id;
      _this.editorLength = _this.editorLength + 1;
      _this.creationEditor({ id: _id, fileName: _fileName, path: path, mode: _mode, encoding: res.encoding, data: res.data, type: _type, historys: res.historys, readOnly: true, historys_file: true, historys_active: obj.history });
      if (callback) callback(res);
    });
  },
  // 打开编辑器文件-方法
  openEditorView: function (path, callback) {
    if (path == undefined) return false;
    // 文件类型（type，列如：JavaScript） 、文件模型（mode，列如：text）、文件标识（id,列如：x8AmsnYn）、文件编号（index,列如：0）、文件路径 (path，列如：/www/root/)
    var _this = this, paths = path.split('/'), _fileName = paths[paths.length - 1], _fileType = this.getFileType(_fileName), _type = _fileType.name, _mode = _fileType.mode, _id = bt.get_random(8), _index = this.editorLength;
    _this.is_file_open(path, function (is_state) {
      if (is_state) {
        $('.ace_conter_menu').find('[title="' + path + '"]').click();
      } else {
        _this.getFileBody({ path: path }, function (res) {
          _this.pathAarry.push(path);
          $('.ace_conter_menu .item').removeClass('active');
          $('.ace_conter_editor .ace_editors').removeClass('active');
          $('.ace_conter_menu').append('<li class="item active item_tab_' + _id + '" title="' + path + '" data-type="' + _type + '" data-mode="' + _mode + '" data-id="' + _id + '" data-fileName="' + _fileName + '">' +
            '<div class="ace_item_box">' +
            '<span class="icon_file"><i class="' + _mode + '-icon"></i></span><span title="' + path + '">' + _fileName + '</span>' +
            '<i class="glyphicon glyphicon-remove icon-tool" aria-hidden="true" data-file-state="0" data-title="' + _fileName + '"></i>' +
            '</div>' +
            '</li>');
          $('.ace_conter_editor').append('<div id="ace_editor_' + _id + '" class="ace_editors active" style="font-size:' + aceEditor.aceConfig.aceEditor.fontSize + 'px"></div>');
          $('[data-menu-path="' + path + '"]').find('.file_fold').addClass('active bg');
          _this.ace_active = _id;
          _this.editorLength = _this.editorLength + 1;
          _this.creationEditor({ id: _id, fileName: _fileName, path: path, mode: _mode, encoding: res.encoding, data: res.data, type: _type, historys: res.historys });
          if (callback) callback(res, _this.editor[_this.ace_active]);
        });
      }
    });
    $('.ace_toolbar_menu').hide();
  },
  // 获取收藏夹列表-方法
  getFavoriteList: function () { },
  // 获取文件列表-请求
  getFileList: function () { },
  // 获取文件内容-请求
  getFileBody: function (obj, callback) {
    var loadT = layer.msg('正在获取文件内容，请稍候...', { time: 0, icon: 16, shade: [0.3, '#000'] }), _this = this;
    $.post("/files?action=GetFileBody", "path=" + encodeURIComponent(obj.path), function (res) {
      layer.close(loadT);
      if (!res.status) {
        if (_this.editorLength == 0) layer.closeAll();
        layer.msg(res.msg, { icon: 2 });

        return false;
      } else {
        if (!aceEditor.isAceView) {
          var _path = obj.path.split('/');
          layer.msg('已打开文件【' + (_path[_path.length - 1]) + '】');
        }
      }
      if (callback) callback(res);
    });
  },
  // 保存文件内容-请求
  saveFileBody: function (obj, success, error) {
    $.ajax({
      type: 'post',
      url: '/files?action=SaveFileBody',
      timeout: 7000, //设置保存超时时间
      data: {
        data: obj.data,
        encoding: obj.encoding.toLowerCase(),
        path: obj.path
      },
      success: function (rdata) {
        if (rdata.status) {
          if (success) success(rdata)
        } else {
          if (error) error(rdata)
        }
        if (!obj.tips) layer.msg(rdata.msg, { icon: rdata.status ? 1 : 2 });
      },
      error: function (err) {
        if (error) error(err)
      }
    });
  },
  // 	保存ace配置
  saveAceConfig: function (data, callback) {
    var loadT = layer.msg('正在设置配置文件，请稍候...', { time: 0, icon: 16, shade: [0.3, '#000'] }), _this = this;
    this.saveFileBody({
      path: '/www/server/panel/BTPanel/static/editor/ace.editor.config.json',
      data: JSON.stringify(data),
      encoding: 'utf-8',
      tips: true,
    }, function (rdata) {
      layer.close(loadT);
      _this.setStorage('aceConfig', JSON.stringify(data));
      if (callback) callback(rdata);
    });
  },
  // 获取配置文件
  getAceConfig: function (callback) {
    var loadT = layer.msg('正在获取配置文件，请稍候...', { time: 0, icon: 16, shade: [0.3, '#000'] }), _this = this;
    this.getFileBody({ path: '/www/server/panel/BTPanel/static/editor/ace.editor.config.json' }, function (rdata) {
      layer.close(loadT);
      _this.setStorage('aceConfig', JSON.stringify(rdata.data));
      if (callback) callback(JSON.parse(rdata.data));
    });
  },
  // 递归保存文件
  saveAllFileBody: function (arry, num, callabck) {
    var _this = this;
    if (typeof num == "function") {
      callabck = num; num = 0;
    } else if (typeof num == "undefined") {
      num = 0;
    }
    if (num == arry.length) {
      if (callabck) callabck();
      layer.msg('全部保存成功', { icon: 1 });
      return false;
    }
    aceEditor.saveFileBody({
      path: arry[num].path,
      data: arry[num].data,
      encoding: arry[num].encoding
    }, function () {
      num = num + 1;
      aceEditor.saveAllFileBody(arry, num, callabck);
    });
  }
}

function openEditorView (type, path, callback) {
  var paths = path.split('/'),
    _fileName = paths[paths.length - 1],
    _aceTmplate = document.getElementById("aceTmplate").innerHTML;
  _aceTmplate = _aceTmplate.replace(/\<\\\/script\>/g, '</script>');
  if (aceEditor.editor !== null) {
    if (aceEditor.isAceView == false) {
      aceEditor.isAceView = true;
      $('.aceEditors .layui-layer-max').click();
    }
    aceEditor.openEditorView(path);
    return false;
  }
  var r = layer.open({
    type: 1,
    maxmin: true,
    shade: false,
    anim:-1,
    area: ['80%', '80%'],
    title: "在线文本编辑器",
    skin: 'aceEditors',
    zIndex: 19999,
    content: _aceTmplate,
    success: function (layero, index) {
      function set_edit_file () {
        aceEditor.ace_active = '';
        aceEditor.eventEditor();
        ace.require("/ace/ext/language_tools");
        ace.config.set("modePath", "/static/editor");
        ace.config.set("workerPath", "/static/editor");
        ace.config.set("themePath", "/static/editor");
        aceEditor.openEditorView(path, callback);
        $('#ace_conter').addClass(aceEditor.aceConfig.aceEditor.editorTheme);
        $('.aceEditors .layui-layer-min').click(function (e) {
          aceEditor.setEditorView();
        });
        $('.aceEditors .layui-layer-max').click(function (e) {
          aceEditor.setEditorView();
        });
      }
      var aceConfig = aceEditor.getStorage('aceConfig');
      if (aceConfig == null) {
        // 获取编辑器配置
        aceEditor.getAceConfig(function (res) {
          aceEditor.aceConfig = res; // 赋值配置参数
          set_edit_file();
        });
      } else {
        aceEditor.aceConfig = JSON.parse(aceConfig);
        typeof aceEditor.aceConfig == 'string' ? aceEditor.aceConfig = JSON.parse(aceEditor.aceConfig) : ''
        set_edit_file();
      }
    },
    cancel: function () {
      for (var item in aceEditor.editor) {
        if (aceEditor.editor[item].fileType == 1) {
          layer.open({
            type: 1,
            area: ['400px', '180px'],
            title: '保存提示',
            content: '<div class="ace-clear-form">\
							<div class="clear-icon"></div>\
							<div class="clear-title">检测到文件未保存，是否保存文件更改？</div>\
							<div class="clear-tips">如果不保存，更改会丢失！</div>\
							<div class="ace-clear-btn" style="">\
								<button type="button" class="btn btn-sm btn-default" style="float:left" data-type="2">不保存文件</button>\
								<button type="button" class="btn btn-sm btn-default" style="margin-right:10px;" data-type="1">取消</button>\
								<button type="button" class="btn btn-sm btn-success" data-type="0">保存文件</button>\
							</div>\
						</div>',
            success: function (layers, indexs) {
              $('.ace-clear-btn button').click(function () {
                var _type = $(this).attr('data-type');
                switch (_type) {
                  case '2':
                    aceEditor.editor = null;
                    aceEditor.editorLength = 0;
                    aceEditor.pathAarry = [];
                    layer.closeAll();
                    break;
                  case '1':
                    layer.close(indexs);
                    break;
                  case '0':
                    var _arry = [], editor = aceEditor['editor'];
                    for (var item in editor) {
                      _arry.push({
                        path: editor[item]['path'],
                        data: editor[item]['ace'].getValue(),
                        encoding: editor[item]['encoding'],
                      })
                    }
                    aceEditor.saveAllFileBody(_arry, function () {
                      $('.ace_conter_menu>.item').each(function (el, indexx) {
                        var _id = $(this).attr('data-id');
                        $(this).find('i').removeClass('glyphicon-exclamation-sign').addClass('glyphicon-remove').attr('data-file-state', '0')
                        aceEditor.editor[_id].fileType = 0;
                      });
                      aceEditor.editor = null;
                      aceEditor.pathAarry = [];
                      layer.closeAll();
                    });
                    break;
                }
              });
            }
          });
          return false;
        }
      }
    },
    end: function () {
      aceEditor.ace_active = '';
      aceEditor.editor = null;
      aceEditor.pathAarry = [];
      aceEditor.menu_path = '';
    }
  });
}


/**
 * AES加密
 * @param {string} s_text 等待加密的字符串
 * @param {string} s_key 16位密钥
 * @param {array} ctx 可选，默认为 { mode: CryptoJS.mode.ECB,padding: CryptoJS.pad.ZeroPadding }
 * @return {string} 
 */
function aes_encrypt (s_text, s_key, ctx) {
  if (ctx == undefined) ctx = { mode: CryptoJS.mode.ECB, padding: CryptoJS.pad.ZeroPadding }
  var key = CryptoJS.enc.Utf8.parse(s_key);
  var encrypt_data = CryptoJS.AES.encrypt(s_text, key, ctx);
  return encrypt_data.toString();
}

/**
 * AES解密
 * @param {string} s_text 等待解密的密文
 * @param {string} s_key 16位密钥
 * @param {array} ctx 可选，默认为 { mode: CryptoJS.mode.ECB,padding: CryptoJS.pad.ZeroPadding }
 * @return {string}
 */
function aes_decrypt (s_text, s_key, ctx) {
  if (ctx == undefined) ctx = { mode: CryptoJS.mode.ECB, padding: CryptoJS.pad.ZeroPadding }
  var key = CryptoJS.enc.Utf8.parse(s_key);
  var decrypt_data = CryptoJS.AES.decrypt(s_text, key, ctx);
  return decrypt_data.toString(CryptoJS.enc.Utf8);
}

/**
 * ajax内容解密
 * @param {string} data 加密的响应数据
 * @param {string} stype ajax中定义的数据类型
 * @return {string} 解密后的响应数据
 */
function ajax_decrypt (data, stype) {
  if (!data) return data;
  if (data.substring(0, 6) == "BT-CRT") {
    var token = $("#request_token_head").attr("token")
    var pwd = token.substring(0, 8) + token.substring(40, 48)
    data = aes_decrypt(data.substring(6), pwd);
    if (stype == undefined) {
      stype = '';
    }
    if (stype.toLowerCase() != 'json') {
      data = JSON.parse(data);
    }
  }
  return data
}
/**
 * 格式化form_data数据，并加密
 * @param {string} form_data 加密前的form_data数据
 * @return {string} 加密后的form_data数据
 */
function format_form_data (form_data) {
  var data_tmp = form_data.split('&');
  var form_info = {}
  var token = $("#request_token_head").attr("token")
  if (!token) return form_data;
  var pwd = token.substring(0, 8) + token.substring(40, 48)
  for (var i = 0; i < data_tmp.length; i++) {
    var tmp = data_tmp[i].split('=');
    if (tmp.length < 2) continue;
    var val = decodeURIComponent(tmp[1].replace(/\+/g, '%20'));
    if (val.length > 3) {
      form_info[tmp[0]] = 'BT-CRT' + aes_encrypt(val, pwd);
    } else {
      form_info[tmp[0]] = val;
    }

  }
  return $.param(form_info);
}

function ajax_encrypt (request) {
  if (!this.type || !this.data || !this.contentType) return;
  if ($("#panel_debug").attr("data") == 'True') return;
  if ($("#panel_debug").attr("data-pyversion") == '2') return;
  if (this.type == 'POST' && this.data.length > 1) {
    this.data = format_form_data(this.data);
  }
}

var gl_error_body = '';
function ajaxSetup () {
  var my_headers = {};
  var request_token_ele = document.getElementById("request_token_head");
  if (request_token_ele) {
    var request_token = request_token_ele.getAttribute('token');
    if (request_token) {
      my_headers['x-http-token'] = request_token
    }
  }
  request_token_cookie = getCookie('request_token');
  if (request_token_cookie) {
    my_headers['x-cookie-token'] = request_token_cookie
  }

  if (my_headers) {
    $.ajaxSetup({
      headers: my_headers,
      error: function (jqXHR, textStatus, errorThrown) {

        if (!jqXHR.responseText) return;
        //会话失效时自动跳转到登录页面
        if (typeof (jqXHR.responseText) == 'string') {
          if ((jqXHR.responseText.indexOf('/static/favicon.ico') != -1 && jqXHR.responseText.indexOf('/static/img/qrCode.png') != -1) || jqXHR.responseText.indexOf('<!DOCTYPE html>') === 0) {
            window.location.href = "/login"
            return
          }
        }
        if (typeof (String.prototype.trim) === "undefined") {
          String.prototype.trim = function () {
            return String(this).replace(/^\s+|\s+$/g, '');
          };
        }

        error_key = 'We need to make sure this has a favicon so that the debugger does';
        error_find = jqXHR.responseText.indexOf(error_key)
        console.log(jqXHR.responseText)
        if (jqXHR.status == 500 && jqXHR.responseText.indexOf('运行时发生错误') != -1) {
          if (jqXHR.responseText.indexOf('请先绑定宝塔帐号!') != -1) {
            if ($('.libLogin').length > 0 || $('.radio_account_view').length > 0) return false;
            bt.pub.bind_btname(function () {
              window.location.reload();
            });
            return;
          }
          gl_error_body = jqXHR.responseText;
          if (jqXHR.responseText.indexOf('建议按顺序逐一尝试以下解决方案') != -1){
            error_msg = jqXHR.responseText.split('Error: ')[1].split("</pre>")[0].replace("面板运行时发生错误:",'').replace("public.PanelError:",'').trim();
          }else{
            error_msg = '<h3>' + jqXHR.responseText.split('<h3>')[1].split('</h3>')[0] + '</h3>'
            error_msg += '<a style="color:dimgrey;font-size:none">' + jqXHR.responseText.split('<h4 style="font-size: none;">')[1].split("</h4>")[0].replace("面板运行时发生错误:",'').replace("public.PanelError:",'').trim() + '</a>';
          }
          
          error_msg += "<br><a class='btlink' onclick='show_error_message()'> >>点击查看详情</a>";
          $(".layui-layer-padding").parents('.layer-anim').remove();
          $(".layui-layer-shade").remove();
          setTimeout(function () {
            layer.open({
              title: false,
              content: error_msg,
              closeBtn: 2,
              btn: false,
              shadeClose: false,
              shade: 0.3,
              icon:2,
              area:"600px",
              success: function () {
                $('pre').scrollTop(100000000000)
              }
            });
          }, 100)
        }
      }
    });
  }
}
ajaxSetup();


function show_error_message() {
    if (error_find != -1) {
        var error_body = gl_error_body.split('<!--')[2].replace('-->', '')
        var tmp = error_body.split('During handling of the above exception, another exception occurred:')
        error_body = tmp[tmp.length - 1];
        var error_msg = '<div>\
        <h3 style="margin-bottom: 10px;">出错了，面板运行时发生错误！</h3>\
        <pre style="height:635px;word-wrap: break-word;white-space: pre-wrap;margin: 0 0 0px">'+ error_body.trim() + '</pre>\
        <ul class="help-info-text">\
          <li style="list-style: none;"><b>很抱歉，面板运行时意外发生错误，请尝试按以下顺序尝试解除此错误：</b></li>\
          <li style="list-style: none;">1、在[首页]右上角点击修复面板，并退出面板重新登录。</li>\
          <li style="list-style: none;">2、如上述尝试未能解除此错误，请截图此窗口到宝塔论坛发贴寻求帮助, 论坛地址：<a class="btlink" href="https://www.bt.cn/bbs" target="_blank">https://www.bt.cn/bbs</a></li>\
        </ul>\
      </div>'

    } else {
        var error_msg = gl_error_body;
    }
    $(".layui-layer-padding").parents('.layer-anim').remove();
    $(".layui-layer-shade").remove();
    setTimeout(function () {
      layer.open({
        title: false,
        content: error_msg,
        closeBtn: 2,
        area: ["1200px", "810px"],
        btn: false,
        shadeClose: false,
        shade: 0.3,
        success: function () {
          $('pre').scrollTop(100000000000)
        }
      });
    }, 100)
}

function RandomStrPwd (b) {
  b = b || 32;
  var c = "AaBbCcDdEeFfGHhiJjKkLMmNnPpRSrTsWtXwYxZyz2345678";
  var a = c.length;
  while (true) {
    var d = "";
    for (i = 0; i < b; i++) {
      d += c.charAt(Math.floor(Math.random() * a))
    }

    if (/^(?:(?=.*[A-Z])(?=.*[a-z])(?=.*[0-9]))/.test(d)) {
      return d;
    }
  }
}

function repeatPwd (a) {
  $("#MyPassword").val(RandomStrPwd(a))
}

function refresh () {
  window.location.reload()
}

function GetBakPost (b) {
  $(".baktext").hide().prev().show();
  var c = $(".baktext").attr("data-id");
  var a = $(".baktext").val();
  if (a == "") {
    a = lan.bt.empty;
  }
  setWebPs(b, c, a);
  $("a[data-id='" + c + "']").html(a);
  $(".baktext").remove()
}

function setWebPs (b, e, a) {
  var d = layer.load({
    shade: true,
    shadeClose: false
  });
  var c = "ps=" + a;
  $.post("/data?action=setPs", "table=" + b + "&id=" + e + "&" + c, function (f) {
    if (f == true) {
      if (b == "sites") {
        getWeb(1)
      } else {
        if (b == "ftps") {
          getFtp(1)
        } else {
          getData(1)
        }
      }
      layer.closeAll();
      layer.msg(lan.public.edit_ok, {
        icon: 1
      });
    } else {
      layer.msg(lan.public.edit_err, {
        icon: 2
      });
      layer.closeAll();
    }
  });
}

$(".menu-icon").click(function () {
  $(".sidebar-scroll").toggleClass("sidebar-close");
  $(".main-content").toggleClass("main-content-open");
  if ($(".sidebar-close")) {
    $(".sub-menu").find(".sub").css("display", "none")
  }
});
var Upload, percentage;

Date.prototype.format = function (b) {
  var c = {
    "M+": this.getMonth() + 1,
    "d+": this.getDate(),
    "h+": this.getHours(),
    "m+": this.getMinutes(),
    "s+": this.getSeconds(),
    "q+": Math.floor((this.getMonth() + 3) / 3),
    S: this.getMilliseconds()
  };
  if (/(y+)/.test(b)) {
    b = b.replace(RegExp.$1, (this.getFullYear() + "").substr(4 - RegExp.$1.length))
  }
  for (var a in c) {
    if (new RegExp("(" + a + ")").test(b)) {
      b = b.replace(RegExp.$1, RegExp.$1.length == 1 ? c[a] : ("00" + c[a]).substr(("" + c[a]).length))
    }
  }
  return b
};

function getLocalTime (a) {
  a = a.toString();
  if (a.length > 10) {
    a = a.substring(0, 10)
  }
  return new Date(parseInt(a) * 1000).format("yyyy/MM/dd hh:mm:ss")
}

function ToSize (a) {
  var d = [" B", " KB", " MB", " GB", " TB", " PB"];
  var e = 1024;
  for (var b = 0; b < d.length; b++) {
    if (a < e) {
      return (b == 0 ? a : a.toFixed(2)) + d[b]
    }
    a /= e
  }
}


function ChangePath (d) {
  setCookie("SetId", d);
  setCookie("SetName", "");
  var c = layer.open({
    type: 1,
    area: "650px",
    title: lan.bt.dir,
    closeBtn: 2,
    shift: 5,
    shadeClose: false,
    content: "<div class='changepath'><div class='path-top'><button type='button' class='btn btn-default btn-sm' onclick='BackFile()'><span class='glyphicon glyphicon-share-alt'></span> " + lan.public.return + "</button><div class='place' id='PathPlace'>" + lan.bt.path + "：<span></span></div></div><div class='path-con'><div class='path-con-left'><dl><dt id='changecomlist' onclick='BackMyComputer()'>" + lan.bt.comp + "</dt></dl></div><div class='path-con-right'><ul class='default' id='computerDefautl'></ul><div class='file-list divtable'><table class='table table-hover' style='border:0 none'><thead><tr class='file-list-head'><th width='40%'>" + lan.bt.filename + "</th><th width='20%'>" + lan.bt.etime + "</th><th width='10%'>" + lan.bt.access + "</th><th width='10%'>" + lan.bt.own + "</th><th width='10%'></th></tr></thead><tbody id='tbody' class='list-list'></tbody></table></div></div></div></div><div class='getfile-btn' style='margin-top:0'><button type='button' class='btn btn-default btn-sm pull-left' onclick='CreateFolder()'>" + lan.bt.adddir + "</button><button type='button' class='btn btn-danger btn-sm mr5' onclick=\"layer.close(getCookie('ChangePath'))\">" + lan.public.close + "</button> <button type='button' class='btn btn-success btn-sm' onclick='GetfilePath()'>" + lan.bt.path_ok + "</button></div>"
  });
  setCookie("ChangePath", c);
  var b = $("#" + d).val();
  tmp = b.split(".");
  if (tmp[tmp.length - 1] == "gz") {
    tmp = b.split("/");
    b = "";
    for (var a = 0; a < tmp.length - 1; a++) {
      b += "/" + tmp[a]
    }
    setCookie("SetName", tmp[tmp.length - 1])
  }
  b = b.replace(/\/\//g, "/");
  GetDiskList(b);
  ActiveDisk()
}

function GetDiskList (b) {
  var d = "";
  var a = "";
  var c = "path=" + b + "&disk=True&showRow=500";
  $.post("/files?action=GetDir", c, function (h) {
    if (h.DISK != undefined) {
      for (var f = 0; f < h.DISK.length; f++) {
        a += "<dd onclick=\"GetDiskList('" + h.DISK[f].path + "')\"><span class='glyphicon glyphicon-hdd'></span>&nbsp;<span>" + h.DISK[f].path + "</span></div></dd>"
      }
      $("#changecomlist").html(a)
    }
    for (var f = 0; f < h.DIR.length; f++) {
      var g = h.DIR[f].split(";");
      var e = g[0];
      if (e.length > 20) {
        e = e.substring(0, 20) + "..."
      }
      if (isChineseChar(e)) {
        if (e.length > 10) {
          e = e.substring(0, 10) + "..."
        }
      }
      d += "<tr><td onclick=\"GetDiskList('" + h.PATH + "/" + g[0] + "')\" title='" + g[0] + "'><span class='glyphicon glyphicon-folder-open'></span>" + e + "</td><td>" + getLocalTime(g[2]) + "</td><td>" + g[3] + "</td><td>" + g[4] + "</td><td><span class='delfile-btn' onclick=\"NewDelFile('" + h.PATH + "/" + g[0] + "')\">X</span></td></tr>"
    }
    if (h.FILES != null && h.FILES != "") {
      for (var f = 0; f < h.FILES.length; f++) {
        var g = h.FILES[f].split(";");
        var e = g[0];
        if (e.length > 20) {
          e = e.substring(0, 20) + "..."
        }
        if (isChineseChar(e)) {
          if (e.length > 10) {
            e = e.substring(0, 10) + "..."
          }
        }
        d += "<tr><td title='" + g[0] + "'><span class='glyphicon glyphicon-file'></span><span>" + e + "</span></td><td>" + getLocalTime(g[2]) + "</td><td>" + g[3] + "</td><td>" + g[4] + "</td><td></td></tr>"
      }
    }
    $(".default").hide();
    $(".file-list").show();
    $("#tbody").html(d);
    if (h.PATH.substr(h.PATH.length - 1, 1) != "/") {
      h.PATH += "/"
    }
    $("#PathPlace").find("span").html(h.PATH);
    ActiveDisk();
    return
  })
}

function CreateFolder () {
  var a = "<tr><td colspan='2'><span class='glyphicon glyphicon-folder-open'></span> <input id='newFolderName' class='newFolderName' type='text' value=''></td><td colspan='3'><button id='nameOk' type='button' class='btn btn-success btn-sm'>" + lan.public.ok + "</button>&nbsp;&nbsp;<button id='nameNOk' type='button' class='btn btn-default btn-sm'>" + lan.public.cancel + "</button></td></tr>";
  if ($("#tbody tr").length == 0) {
    $("#tbody").append(a)
  } else {
    $("#tbody tr:first-child").before(a)
  }
  $(".newFolderName").focus();
  $("#nameOk").click(function () {
    var c = $("#newFolderName").val();
    var b = $("#PathPlace").find("span").text();
    newTxt = b.replace(new RegExp(/(\/\/)/g), "/") + c;
    var d = "path=" + newTxt;
    $.post("/files?action=CreateDir", d, function (e) {
      if (e.status == true) {
        layer.msg(e.msg, {
          icon: 1
        })
      } else {
        layer.msg(e.msg, {
          icon: 2
        })
      }
      GetDiskList(b)
    })
  });
  $("#nameNOk").click(function () {
    $(this).parents("tr").remove()
  })
}

function NewDelFile (c) {
  var a = $("#PathPlace").find("span").text();
  newTxt = c.replace(new RegExp(/(\/\/)/g), "/");
  var b = "path=" + newTxt + "&empty=True";
  $.post("/files?action=DeleteDir", b, function (d) {
    if (d.status == true) {
      layer.msg(d.msg, {
        icon: 1
      })
    } else {
      layer.msg(d.msg, {
        icon: 2
      })
    }
    this.get_file_list(a);
  })
}

function ActiveDisk () {
  var a = $("#PathPlace").find("span").text().substring(0, 1);
  switch (a) {
    case "C":
      $(".path-con-left dd:nth-of-type(1)").css("background", "#eee").siblings().removeAttr("style");
      break;
    case "D":
      $(".path-con-left dd:nth-of-type(2)").css("background", "#eee").siblings().removeAttr("style");
      break;
    case "E":
      $(".path-con-left dd:nth-of-type(3)").css("background", "#eee").siblings().removeAttr("style");
      break;
    case "F":
      $(".path-con-left dd:nth-of-type(4)").css("background", "#eee").siblings().removeAttr("style");
      break;
    case "G":
      $(".path-con-left dd:nth-of-type(5)").css("background", "#eee").siblings().removeAttr("style");
      break;
    case "H":
      $(".path-con-left dd:nth-of-type(6)").css("background", "#eee").siblings().removeAttr("style");
      break;
    default:
      $(".path-con-left dd").removeAttr("style")
  }
}

function BackMyComputer () {
  $(".default").show();
  $(".file-list").hide();
  $("#PathPlace").find("span").html("");
  ActiveDisk()
}

function BackFile () {
  var c = $("#PathPlace").find("span").text();
  if (c.substr(c.length - 1, 1) == "/") {
    c = c.substr(0, c.length - 1)
  }
  var d = c.split("/");
  var a = "";
  if (d.length > 1) {
    var e = d.length - 1;
    for (var b = 0; b < e; b++) {
      a += d[b] + "/"
    }
    GetDiskList(a.replace("//", "/"))
  } else {
    a = d[0]
  }
  if (d.length == 1) { }
}

function GetfilePath () {
  var a = $("#PathPlace").find("span").text();
  a = a.replace(new RegExp(/(\\)/g), "/");
  setCookie('path_dir_change', a);
  $("#" + getCookie("SetId")).val(a + getCookie("SetName"));
  layer.close(getCookie("ChangePath"))
}

function setCookie (a, c) {
  var b = 30;
  var d = new Date();
  d.setTime(d.getTime() + b * 24 * 60 * 60 * 1000);
  var is_https = window.location.protocol == 'https:'
  var samesite = ';Secure; Path=/; SameSite=None'
  document.cookie = a + "=" + escape(c) + ";expires=" + d.toGMTString() + (is_https ? samesite : '')
}

function getCookie (b) {
  var a, c = new RegExp("(^| )" + b + "=([^;]*)(;|$)");
  if (a = document.cookie.match(c)) {
    return unescape(a[2])
  } else {
    return null
  }
}

function aotuHeight () {
  var a = $("body").height() - 50;
  $(".main-content").css("min-height", a)
}
$(function () {
  aotuHeight()
});
$(window).resize(function () {
  aotuHeight()
});

function showHidePwd () {
  var a = "glyphicon-eye-open",
    b = "glyphicon-eye-close";
  $(".pw-ico").click(function () {
    var g = $(this).attr("class"),
      e = $(this).prev();
    if (g.indexOf(a) > 0) {
      var h = e.attr("data-pw");
      $(this).removeClass(a).addClass(b);
      e.text(h)
    } else {
      $(this).removeClass(b).addClass(a);
      e.text("**********")
    }
    var d = $(this).next().position().left;
    var f = $(this).next().position().top;
    var c = $(this).next().width();
    $(this).next().next().css({
      left: d + c + "px",
      top: f + "px"
    })
  })
}

function openPath (a) {
  setCookie("Path", a);
  window.location.href = "/files"
}

function OnlineEditFile (k, f) {
  if (k != 0) {
    var l = $("#PathPlace input").val();
    var h = encodeURIComponent($("#textBody").val());
    var a = $("select[name=encoding]").val();
    var loadT = layer.msg(lan.bt.save_file, {
      icon: 16,
      time: 0
    });
    $.post("/files?action=SaveFileBody", "data=" + h + "&path=" + encodeURIComponent(f) + "&encoding=" + a, function (m) {
      if (k == 1) {
        layer.close(loadT);
      }
      layer.msg(m.msg, {
        icon: m.status ? 1 : 2
      });
    });
    return
  }
  var e = layer.msg(lan.bt.read_file, {
    icon: 16,
    time: 0
  });
  var g = f.split(".");
  var b = g[g.length - 1];
  var d;
  switch (b) {
    case "html":
      var j = {
        name: "htmlmixed",
        scriptTypes: [{
          matches: /\/x-handlebars-template|\/x-mustache/i,
          mode: null
        }, {
          matches: /(text|application)\/(x-)?vb(a|script)/i,
          mode: "vbscript"
        }]
      };
      d = j;
      break;
    case "htm":
      var j = {
        name: "htmlmixed",
        scriptTypes: [{
          matches: /\/x-handlebars-template|\/x-mustache/i,
          mode: null
        }, {
          matches: /(text|application)\/(x-)?vb(a|script)/i,
          mode: "vbscript"
        }]
      };
      d = j;
      break;
    case "js":
      d = "text/javascript";
      break;
    case "json":
      d = "application/ld+json";
      break;
    case "css":
      d = "text/css";
      break;
    case "php":
      d = "application/x-httpd-php";
      break;
    case "tpl":
      d = "application/x-httpd-php";
      break;
    case "xml":
      d = "application/xml";
      break;
    case "sql":
      d = "text/x-sql";
      break;
    case "conf":
      d = "text/x-nginx-conf";
      break;
    default:
      var j = {
        name: "htmlmixed",
        scriptTypes: [{
          matches: /\/x-handlebars-template|\/x-mustache/i,
          mode: null
        }, {
          matches: /(text|application)\/(x-)?vb(a|script)/i,
          mode: "vbscript"
        }]
      };
      d = j
  }
  $.post("/files?action=GetFileBody", "path=" + encodeURIComponent(f), function (s) {
    if (s.status === false) {
      layer.msg(s.msg, { icon: 5 });
      return;
    }
    layer.close(e);
    var u = ["utf-8", "GBK", "GB2312", "BIG5"];
    var n = "";
    var m = "";
    var o = "";
    for (var p = 0; p < u.length; p++) {
      m = s.encoding == u[p] ? "selected" : "";
      n += '<option value="' + u[p] + '" ' + m + ">" + u[p] + "</option>"
    }
    var r = layer.open({
      type: 1,
      shift: 5,
      closeBtn: 2,
      area: ["90%", "90%"],
      title: lan.bt.edit_title + "[" + f + "]",
      content: '<form class="bt-form pd20 pb70"><div class="line"><p style="color:red;margin-bottom:10px">' + lan.bt.edit_ps + '			<select class="bt-input-text" name="encoding" style="width: 74px;position: absolute;top: 31px;right: 19px;height: 22px;z-index: 9999;border-radius: 0;">' + n + '</select></p><textarea class="mCustomScrollbar bt-input-text" id="textBody" style="width:100%;margin:0 auto;line-height: 1.8;position: relative;top: 10px;" value="" />			</div>			<div class="bt-form-submit-btn" style="position:absolute; bottom:0; width:100%">			<button type="button" class="btn btn-danger btn-sm btn-editor-close">' + lan.public.close + '</button>			<button id="OnlineEditFileBtn" type="button" class="btn btn-success btn-sm">' + lan.public.save + '</button>			</div>			</form>'
    });
    $("#textBody").text(s.data);
    var q = $(window).height() * 0.9;
    $("#textBody").height(q - 160);
    var t = CodeMirror.fromTextArea(document.getElementById("textBody"), {
      extraKeys: {
        "Ctrl-F": "findPersistent",
        "Ctrl-H": "replaceAll",
        "Ctrl-S": function () {
          $("#textBody").text(t.getValue());
          OnlineEditFile(2, f)
        }
      },
      mode: d,
      lineNumbers: true,
      matchBrackets: true,
      matchtags: true,
      autoMatchParens: true
    });
    t.focus();
    t.setSize("auto", q - 150);
    $("#OnlineEditFileBtn").click(function () {
      $("#textBody").text(t.getValue());
      OnlineEditFile(1, f);
    });
    $(".btn-editor-close").click(function () {
      layer.close(r);
    });
  });
}

function ServiceAdmin (a, b) {
  if (!isNaN(a)) {
    a = "php-fpm-" + a
  }
  a = a.replace('_soft', '');
  var c = "name=" + a + "&type=" + b;
  var d = "";

  switch (b) {
    case "stop":
      d = lan.bt.stop;
      break;
    case "start":
      d = lan.bt.start;
      break;
    case "restart":
      d = lan.bt.restart;
      break;
    case "reload":
      d = lan.bt.reload;
      break
  }
  layer.confirm(lan.get('service_confirm', [d, a]), {
    icon: 3,
    closeBtn: 2
  }, function () {
    var e = layer.msg(lan.get('service_the', [d, a]), {
      icon: 16,
      time: 0
    });
    $.post("/system?action=ServiceAdmin", c, function (g) {
      layer.close(e);

      var f = g.status ? lan.get('service_ok', [a, d]) : lan.get('service_err', [a, d]);
      layer.msg(f, {
        icon: g.status ? 1 : 2
      });
      if (b != "reload" && g.status == true) {
        setTimeout(function () {
          window.location.reload()
        }, 1000)
      }
      if (!g.status) {
        layer.msg(g.msg, {
          icon: 2,
          time: 0,
          shade: 0.3,
          shadeClose: true
        })
      }
    }).error(function () {
      layer.close(e);
      layer.msg(lan.public.success, {
        icon: 1
      })
    })
  })
}

function GetConfigFile (a) {
  var b = "";
  switch (a) {
    case "mysql":
      b = "/etc/my.cnf";
      break;
    case "nginx":
      b = "/www/server/nginx/conf/nginx.conf";
      break;
    case "pure-ftpd":
      b = "/www/server/pure-ftpd/etc/pure-ftpd.conf";
      break;
    case "apache":
      b = "/www/server/apache/conf/httpd.conf";
      break;
    case "tomcat":
      b = "/www/server/tomcat/conf/server.xml";
      break;
    default:
      b = "/www/server/php/" + a + "/etc/php.ini";
      break
  }
  OnlineEditFile(0, b)
}

function GetPHPStatus (a) {
  if (a == "52") {
    layer.msg(lan.bt.php_status_err, {
      icon: 2
    });
    return
  }
  $.post("/ajax?action=GetPHPStatus", "version=" + a, function (b) {
    layer.open({
      type: 1,
      area: "400",
      title: lan.bt.php_status_title,
      closeBtn: 2,
      shift: 5,
      shadeClose: true,
      content: "<div style='margin:15px;'><table class='table table-hover table-bordered'>						<tr><th>" + lan.bt.php_pool + "</th><td>" + b.pool + "</td></tr>						<tr><th>" + lan.bt.php_manager + "</th><td>" + ((b["process manager"] == "dynamic") ? lan.bt.dynamic : lan.bt.static) + "</td></tr>						<tr><th>" + lan.bt.php_start + "</th><td>" + b["start time"] + "</td></tr>						<tr><th>" + lan.bt.php_accepted + "</th><td>" + b["accepted conn"] + "</td></tr>						<tr><th>" + lan.bt.php_queue + "</th><td>" + b["listen queue"] + "</td></tr>						<tr><th>" + lan.bt.php_max_queue + "</th><td>" + b["max listen queue"] + "</td></tr>						<tr><th>" + lan.bt.php_len_queue + "</th><td>" + b["listen queue len"] + "</td></tr>						<tr><th>" + lan.bt.php_idle + "</th><td>" + b["idle processes"] + "</td></tr>						<tr><th>" + lan.bt.php_active + "</th><td>" + b["active processes"] + "</td></tr>						<tr><th>" + lan.bt.php_total + "</th><td>" + b["total processes"] + "</td></tr>						<tr><th>" + lan.bt.php_max_active + "</th><td>" + b["max active processes"] + "</td></tr>						<tr><th>" + lan.bt.php_max_children + "</th><td>" + b["max children reached"] + "</td></tr>						<tr><th>" + lan.bt.php_slow + "</th><td>" + b["slow requests"] + "</td></tr>					 </table></div>"
    })
  })
}

function GetNginxStatus () {
  $.post("/ajax?action=GetNginxStatus", "", function (a) {
    layer.open({
      type: 1,
      area: "400",
      title: lan.bt.nginx_title,
      closeBtn: 2,
      shift: 5,
      shadeClose: true,
      content: "<div style='margin:15px;'><table class='table table-hover table-bordered'>						<tr><th>" + lan.bt.nginx_active + "</th><td>" + a.active + "</td></tr>						<tr><th>" + lan.bt.nginx_accepts + "</th><td>" + a.accepts + "</td></tr>						<tr><th>" + lan.bt.nginx_handled + "</th><td>" + a.handled + "</td></tr>						<tr><th>" + lan.bt.nginx_requests + "</th><td>" + a.requests + "</td></tr>						<tr><th>" + lan.bt.nginx_reading + "</th><td>" + a.Reading + "</td></tr>						<tr><th>" + lan.bt.nginx_writing + "</th><td>" + a.Writing + "</td></tr>						<tr><th>" + lan.bt.nginx_waiting + "</th><td>" + a.Waiting + "</td></tr>					 </table></div>"
    })
  })
}

function divcenter () {
  $(".layui-layer").css("position", "absolute");
  var c = $(window).width();
  var b = $(".layui-layer").outerWidth();
  var g = $(window).height();
  var f = $(".layui-layer").outerHeight();
  var a = (c - b) / 2;
  var e = (g - f) / 2 > 0 ? (g - f) / 2 : 10;
  var d = $(".layui-layer").offset().left - $(".layui-layer").position().left;
  var h = $(".layui-layer").offset().top - $(".layui-layer").position().top;
  a = a + $(window).scrollLeft() - d;
  e = e + $(window).scrollTop() - h;
  $(".layui-layer").css("left", a + "px");
  $(".layui-layer").css("top", e + "px")
}

function isChineseChar (b) {
  var a = /[\u4E00-\u9FA5\uF900-\uFA2D]/;
  return a.test(b)
}

function SafeMessage (j, h, g, f) {
  if (f == undefined) {
    f = ""
  }
  var d = Math.round(Math.random() * 9 + 1);
  var c = Math.round(Math.random() * 9 + 1);
  var e = "";
  e = d + c;
  sumtext = d + " + " + c;
  setCookie("vcodesum", e);
  var mess = layer.open({
    type: 1,
    title: j,
    area: "350px",
    closeBtn: 2,
    shadeClose: true,
    content: "<div class='bt-form webDelete pd20 pb70'><p>" + h + "</p>" + f + "<div class='vcode'>" + lan.bt.cal_msg + "<span class='text'>" + sumtext + "</span>=<input type='number' id='vcodeResult' value=''></div><div class='bt-form-submit-btn'><button type='button' class='btn btn-danger btn-sm bt-cancel'>" + lan.public.cancel + "</button> <button type='button' id='toSubmit' class='btn btn-success btn-sm' >" + lan.public.ok + "</button></div></div>"
  });
  $("#vcodeResult").focus().keyup(function (a) {
    if (a.keyCode == 13) {
      $("#toSubmit").click()
    }
  });
  $(".bt-cancel").click(function () {
    layer.close(mess);
  });
  $("#toSubmit").click(function () {
    var a = $("#vcodeResult").val().replace(/ /g, "");
    if (a == undefined || a == "") {
      layer.msg('请正确输入计算结果!');
      return
    }
    if (a != getCookie("vcodesum")) {
      layer.msg('请正确输入计算结果!');
      return
    }
    layer.close(mess);
    g();
  })
}

$(function () {
  $(".fb-ico").hover(function () {
    $(".fb-text").css({
      left: "36px",
      top: 0,
      width: "80px"
    })
  }, function () {
    $(".fb-text").css({
      left: 0,
      width: "36px"
    })
  }).click(function () {
    $(".fb-text").css({
      left: 0,
      width: "36px"
    });
    $(".zun-feedback-suggestion").show()
  });
  $(".fb-close").click(function () {
    $(".zun-feedback-suggestion").hide()
  });
  $(".fb-attitudes li").click(function () {
    $(this).addClass("fb-selected").siblings().removeClass("fb-selected")
  })
});
$("#dologin").click(function () {
  layer.confirm(lan.bt.loginout, {
    icon: 3,
    closeBtn: 2
  }, function () {
    window.location.href = "/login?dologin=True";
  });
  return false
});

function setPassword (a) {
  if (a == 1) {
    p1 = $("#p1").val();
    p2 = $("#p2").val();
    if (p1 == "" || p1.length < 8) {
      layer.msg(lan.bt.pass_err_len, {
        icon: 2
      });
      return
    }

    //准备弱口令匹配元素
    var checks = ['admin888', '123123123', '12345678', '45678910', '87654321', 'asdfghjkl', 'password', 'qwerqwer'];
    pchecks = 'abcdefghijklmnopqrstuvwxyz1234567890';
    for (var i = 0; i < pchecks.length; i++) {
      checks.push(pchecks[i] + pchecks[i] + pchecks[i] + pchecks[i] + pchecks[i] + pchecks[i] + pchecks[i] + pchecks[i]);
    }

    //检查弱口令
    cps = p1.toLowerCase();
    var isError = "";
    for (var i = 0; i < checks.length; i++) {
      if (cps == checks[i]) {
        isError += '[' + checks[i] + '] ';
      }
    }

    if (isError != "") {
      layer.msg(lan.bt.pass_err + isError, { icon: 5 });
      return;
    }


    if (p1 != p2) {
      layer.msg(lan.bt.pass_err_re, {
        icon: 2
      });
      return
    }
    $.post("/config?action=setPassword", "password1=" + encodeURIComponent(p1) + "&password2=" + encodeURIComponent(p2), function (b) {
      if (b.status) {
        layer.closeAll();
        layer.msg(b.msg, {
          icon: 1,
          time: 1000
        }, function () {
          window.location.href = '/login?dologin=True';
        })
      } else {
        layer.msg(b.msg, {
          icon: 2
        })
      }
    });
    return
  }
  layer.open({
    type: 1,
    area: "290px",
    title: lan.bt.pass_title,
    closeBtn: 2,
    shift: 5,
    shadeClose: false,
    content: "<div class='bt-form pd20 pb70'><div class='line'><span class='tname'>" + lan.public.pass + "</span><div class='info-r'><input class='bt-input-text' type='text' name='password1' id='p1' value='' placeholder='" + lan.bt.pass_new_title + "' style='width:100%'/></div></div><div class='line'><span class='tname'>" + lan.bt.pass_re + "</span><div class='info-r'><input class='bt-input-text' type='text' name='password2' id='p2' value='' placeholder='" + lan.bt.pass_re_title + "' style='width:100%' /></div></div><div class='bt-form-submit-btn'><span style='float: left;' title='" + lan.bt.pass_rep + "' class='btn btn-default btn-sm' onclick='randPwd(10)'>" + lan.bt.pass_rep_btn + "</span><button type='button' class='btn btn-danger btn-sm' onclick=\"layer.closeAll()\">" + lan.public.close + "</button> <button type='button' class='btn btn-success btn-sm' onclick=\"setPassword(1)\">" + lan.public.edit + "</button></div></div>"
  });
}


function randPwd (len) {
  if (len == undefined) len = 12;
  var pwd = RandomStrPwd(len);
  $("#p1").val(pwd);
  $("#p2").val(pwd);
  layer.msg(lan.bt.pass_rep_ps, { time: 2000 })
}

function setUserName (a) {
  if (a == 1) {
    p1 = $("#p1").val();
    p2 = $("#p2").val();
    if (p1 == "" || p1.length < 3) {
      layer.msg(lan.bt.user_len, {
        icon: 2
      });
      return
    }
    if (p1 != p2) {
      layer.msg(lan.bt.user_err_re, {
        icon: 2
      });
      return
    }
    var checks = ['admin', 'root', 'admin123', '123456'];

    if ($.inArray(p1, checks) >= 0) {
      layer.msg('禁止使用常用用户名', {
        icon: 2
      });
      return;
    }

    $.post("/config?action=setUsername", "username1=" + encodeURIComponent(p1) + "&username2=" + encodeURIComponent(p2), function (b) {
      if (b.status) {
        layer.closeAll();
        layer.msg(b.msg, {
          icon: 1,
          time: 1000
        }, function () {
          window.location.href = '/login?dologin=True';
        });
        $("input[name='username_']").val(p1)
      } else {
        layer.msg(b.msg, {
          icon: 2
        })
      }
    });
    return
  }
  layer.open({
    type: 1,
    area: "290px",
    title: lan.bt.user_title,
    closeBtn: 2,
    shift: 5,
    shadeClose: false,
    content: "<div class='bt-form pd20 pb70'><div class='line'><span class='tname'>" + lan.bt.user + "</span><div class='info-r'><input class='bt-input-text' type='text' name='password1' id='p1' value='' placeholder='" + lan.bt.user_new + "' style='width:100%'/></div></div><div class='line'><span class='tname'>" + lan.bt.pass_re + "</span><div class='info-r'><input class='bt-input-text' type='text' name='password2' id='p2' value='' placeholder='" + lan.bt.pass_re_title + "' style='width:100%'/></div></div><div class='bt-form-submit-btn'><button type='button' class='btn btn-danger btn-sm' onclick=\"layer.closeAll()\">" + lan.public.close + "</button> <button type='button' class='btn btn-success btn-sm' onclick=\"setUserName(1)\">" + lan.public.edit + "</button></div></div>"
  })
}
var openWindow = null;
var downLoad = null;
var speed = null;

function task () {
  messagebox();
}

function ActionTask () {
  var a = layer.msg(lan.public.the_del, {
    icon: 16,
    time: 0,
    shade: [0.3, "#000"]
  });
  $.post("/files?action=ActionTask", "", function (b) {
    layer.close(a);
    layer.msg(b.msg, {
      icon: b.status ? 1 : 5
    })
  })
}

function RemoveTask (id) {
  var loadT = bt.load(lan.public.the_del);
  bt.send('RemoveTask', 'files/RemoveTask', { id: id }, function (res) {
    bt.msg(res)
    reader_realtime_tasks()
  })
}


function GetTaskCount () {
  $.post("/ajax?action=GetTaskCount", "", function (a) {
    if (a.status === false) {
      window.location.href = '/login?dologin=True';
      return;
    }
    $(".task").text(a)
  })
}

function setSelectChecked (c, d) {
  var a = document.getElementById(c);
  for (var b = 0; b < a.options.length; b++) {
    if (a.options[b].innerHTML == d) {
      a.options[b].selected = true;
      break
    }
  }
}
GetTaskCount();
function RecInstall () {
  $.post("/ajax?action=GetSoftList", "", function (l) {
    var c = "";
    var g = "";
    var e = "";
    for (var h = 0; h < l.length; h++) {
      if (l[h].name == "Tomcat") {
        continue
      }
      var o = "";
      var m = "<input id='data_" + l[h].name + "' data-info='" + l[h].name + " " + l[h].versions[0].version + "' type='checkbox' checked>";
      for (var b = 0; b < l[h].versions.length; b++) {
        var d = "";
        if ((l[h].name == "PHP" && (l[h].versions[b].version == "5.4" || l[h].versions[b].version == "54")) || (l[h].name == "MySQL" && l[h].versions[b].version == "5.5") || (l[h].name == "phpMyAdmin" && l[h].versions[b].version == "4.4")) {
          d = "selected";
          m = "<input id='data_" + l[h].name + "' data-info='" + l[h].name + " " + l[h].versions[b].version + "' type='checkbox' checked>"
        }
        o += "<option value='" + l[h].versions[b].version + "' " + d + ">" + l[h].name + " " + l[h].versions[b].version + "</option>"
      }
      var f = "<li><span class='ico'><img src='/static/img/" + l[h].name.toLowerCase() + ".png'></span><span class='name'><select id='select_" + l[h].name + "' class='sl-s-info'>" + o + "</select></span><span class='pull-right'>" + m + "</span></li>";
      if (l[h].name == "Nginx") {
        c = f
      } else {
        if (l[h].name == "Apache") {
          g = f
        } else {
          e += f
        }
      }
    }
    c += e;
    g += e;
    g = g.replace(new RegExp(/(data_)/g), "apache_").replace(new RegExp(/(select_)/g), "apache_select_");
    var k = layer.open({
      type: 1,
      title: lan.bt.install_title,
      area: ["658px", "423px"],
      closeBtn: 2,
      shadeClose: false,
      content: "<div class='rec-install'><div class='important-title'><p><span class='glyphicon glyphicon-alert' style='color: #f39c12; margin-right: 10px;'></span>" + lan.bt.install_ps + " <a href='javascript:jump()' style='color:#20a53a'>" + lan.bt.install_s + "</a> " + lan.bt.install_s1 + "</p></div><div class='rec-box'><h3>" + lan.bt.install_lnmp + "</h3><div class='rec-box-con'><ul class='rec-list'>" + c + "</ul><p class='fangshi'>" + lan.bt.install_type + "：<label data-title='" + lan.bt.install_rpm_title + "' style='margin-right:0'>" + lan.bt.install_rpm + "<input type='checkbox' checked></label><label data-title='" + lan.bt.install_src_title + "'>" + lan.bt.install_src + "<input type='checkbox'></label></p><div class='onekey'>" + lan.bt.install_key + "</div></div></div><div class='rec-box' style='margin-left:16px'><h3>LAMP</h3><div class='rec-box-con'><ul class='rec-list'>" + g + "</ul><p class='fangshi'>" + lan.bt.install_type + "：<label data-title='" + lan.bt.install_rpm_title + "' style='margin-right:0'>" + lan.bt.install_rpm + "<input type='checkbox' checked></label><label data-title='" + lan.bt.install_src_title + "'>" + lan.bt.install_src + "<input type='checkbox'></label></p><div class='onekey'>一键安装</div></div></div></div>"
    });
    $(".fangshi input").click(function () {
      $(this).attr("checked", "checked").parent().siblings().find("input").removeAttr("checked")
    });
    $(".sl-s-info").change(function () {
      var p = $(this).find("option:selected").text();
      var n = $(this).attr("id");
      p = p.toLowerCase();
      $(this).parents("li").find("input").attr("data-info", p)
    });
    $("#apache_select_PHP").change(function () {
      var n = $(this).val();
      j(n, "apache_select_", "apache_")
    });
    $("#select_PHP").change(function () {
      var n = $(this).val();
      j(n, "select_", "data_")
    });

    function j (p, r, q) {
      var n = "4.4";
      switch (p) {
        case "5.2":
          n = "4.0";
          break;
        case "5.3":
          n = "4.0";
          break;
        case "5.4":
          n = "4.4";
          break;
        case "5.5":
          n = "4.4";
          break;
        default:
          n = "4.7"
      }
      $("#" + r + "phpMyAdmin option[value='" + n + "']").attr("selected", "selected").siblings().removeAttr("selected");
      $("#" + r + "_phpMyAdmin").attr("data-info", "phpmyadmin " + n)
    }
    $("#select_MySQL,#apache_select_MySQL").change(function () {
      var n = $(this).val();
      a(n)
    });

    $("#apache_select_Apache").change(function () {
      var apacheVersion = $(this).val();
      if (apacheVersion == '2.2') {
        layer.msg(lan.bt.install_apache22);
      } else {
        layer.msg(lan.bt.install_apache24);
      }
    });

    $("#apache_select_PHP").change(function () {
      var apacheVersion = $("#apache_select_Apache").val();
      var phpVersion = $(this).val();
      if (apacheVersion == '2.2') {
        if (phpVersion != '5.2' && phpVersion != '5.3' && phpVersion != '5.4') {
          layer.msg(lan.bt.insatll_s22 + 'PHP-' + phpVersion, { icon: 5 });
          $(this).val("5.4");
          $("#apache_PHP").attr('data-info', 'php 5.4');
          return false;
        }
      } else {
        if (phpVersion == '5.2') {
          layer.msg(lan.bt.insatll_s24 + 'PHP-' + phpVersion, { icon: 5 });
          $(this).val("5.4");
          $("#apache_PHP").attr('data-info', 'php 5.4');
          return false;
        }
      }
    });

    function a (n) {
      memSize = getCookie("memSize");
      max = 64;
      msg = "64M";
      switch (n) {
        case "5.1":
          max = 256;
          msg = "256M";
          break;
        case "5.7":
          max = 1500;
          msg = "2GB";
          break;
        case "5.6":
          max = 800;
          msg = "1GB";
          break;
        case "AliSQL":
          max = 800;
          msg = "1GB";
          break;
        case "mariadb_10.0":
          max = 800;
          msg = "1GB";
          break;
        case "mariadb_10.1":
          max = 1500;
          msg = "2GB";
          break
      }
      if (memSize < max) {
        layer.msg(lan.bt.insatll_mem.replace("{1}", msg).replace("{2}", n), {
          icon: 5
        })
      }
    }
    var de = null;
    $(".onekey").click(function () {
      if (de) return;
      var v = $(this).prev().find("input").eq(0).prop("checked") ? "1" : "0";
      var r = $(this).parents(".rec-box-con").find(".rec-list li").length;
      var n = "";
      var q = "";
      var p = "";
      var x = "";
      var s = "";
      de = true;
      for (var t = 0; t < r; t++) {
        var w = $(this).parents(".rec-box-con").find("ul li").eq(t);
        var u = w.find("input");
        if (u.prop("checked")) {
          n += u.attr("data-info") + ","
        }
      }
      q = n.split(",");
      loadT = layer.msg(lan.bt.install_to, {
        icon: 16,
        time: 0,
        shade: [0.3, "#000"]
      });
      for (var t = 0; t < q.length - 1; t++) {
        p = q[t].split(" ")[0].toLowerCase();
        x = q[t].split(" ")[1];
        s = "name=" + p + "&version=" + x + "&type=" + v + "&id=" + (t + 1);
        $.ajax({
          url: "/files?action=InstallSoft",
          data: s,
          type: "POST",
          async: false,
          success: function (y) { }
        });
      }
      layer.close(loadT);
      layer.close(k);
      setTimeout(function () {
        GetTaskCount()
      }, 2000);
      layer.msg(lan.bt.install_ok, {
        icon: 1
      });
      setTimeout(function () {
        task()
      }, 1000)
    });
    InstallTips();
    fly("onekey")
  })
}

function jump () {
  layer.closeAll();
  window.location.href = "/soft"
}

function InstallTips () {
  $(".fangshi label").mouseover(function () {
    var a = $(this).attr("data-title");
    layer.tips(a, this, {
      tips: [1, "#787878"],
      time: 0
    })
  }).mouseout(function () {
    $(".layui-layer-tips").remove()
  })
}

function fly (a) {
  var b = $("#task").offset();
  $("." + a).click(function (d) {
    var e = $(this);
    var c = $('<span class="yuandian"></span>');
    c.fly({
      start: {
        left: d.pageX,
        top: d.pageY
      },
      end: {
        left: b.left + 10,
        top: b.top + 10,
        width: 0,
        height: 0
      },
      onEnd: function () {
        layer.closeAll();
        layer.msg(lan.bt.task_add, {
          icon: 1
        });
        GetTaskCount()
      }
    });
  });
};


//检查选中项
function checkSelect () {
  setTimeout(function () {
    var checkList = $("input[name=id]");
    var count = 0;
    for (var i = 0; i < checkList.length; i++) {
      if (checkList[i].checked) count++;
    }
    if (count > 0) {
      $("#allDelete").show();
    } else {
      $("#allDelete").hide();
    }
  }, 5);
}

//处理排序
function listOrder (skey, type, obj) {
  or = getCookie('order');
  orderType = 'desc';
  if (or) {
    if (or.split(' ')[1] == 'desc') {
      orderType = 'asc';
    }
  }

  setCookie('order', skey + ' ' + orderType);

  switch (type) {
    case 'site':
      getWeb(1);
      break;
    case 'database':
      getData(1);
      break;
    case 'ftp':
      getFtp(1);
      break;
  }
  $(obj).find(".glyphicon-triangle-bottom").remove();
  $(obj).find(".glyphicon-triangle-top").remove();
  if (orderType == 'asc') {
    $(obj).append("<span class='glyphicon glyphicon-triangle-bottom' style='margin-left:5px;color:#bbb'></span>");
  } else {
    $(obj).append("<span class='glyphicon glyphicon-triangle-top' style='margin-left:5px;color:#bbb'></span>");
  }
}

// //去关联列表
// function GetBtpanelList(){
// 	var con ='';
// 	$.post("/config?action=GetPanelList",function(rdata){
// 		for(var i=0; i<rdata.length; i++){
// 			con +='<h3 class="mypcip mypcipnew" style="opacity:.6" data-url="'+rdata[i].url+'" data-user="'+rdata[i].username+'" data-pw="'+rdata[i].password+'"><span class="f14 cw">'+rdata[i].title+'</span><em class="btedit" onclick="bindBTPanel(0,\'c\',\''+rdata[i].title+'\',\''+rdata[i].id+'\',\''+rdata[i].url+'\',\''+rdata[i].username+'\',\''+rdata[i].password+'\')"></em></h3>'
// 		}
// 		$("#newbtpc").html(con);
// 		$(".mypcipnew").hover(function(){
// 			$(this).css("opacity","1");
// 		},function(){
// 			$(this).css("opacity",".6");
// 		}).click(function(){
// 		$("#btpanelform").remove();
// 		var murl = $(this).attr("data-url");
// 		var user = $(this).attr("data-user");
// 		var pw = $(this).attr("data-pw");
// 		layer.open({
// 		  type: 2,
// 		  title: false,
// 		  closeBtn: 0, //不显示关闭按钮
// 		  shade: [0],
// 		  area: ['340px', '215px'],
// 		  offset: 'rb', //右下角弹出
// 		  time: 5, //2秒后自动关闭
// 		  anim: 2,
// 		  content: [murl+'/login', 'no']
// 		});
// 			var loginForm ='<div id="btpanelform" style="display:none"><form id="toBtpanel" action="'+murl+'/login" method="post" target="btpfrom">\
// 				<input name="username" id="btp_username" value="'+user+'" type="text">\
// 				<input name="password" id="btp_password" value="'+pw+'" type="password">\
// 				<input name="code" id="bt_code" value="12345" type="text">\
// 			</form><iframe name="btpfrom" src=""></iframe></div>';
// 			$("body").append(loginForm);
// 			layer.msg(lan.bt.panel_open,{icon:16,shade: [0.3, '#000'],time:1000});
// 			setTimeout(function(){
// 				$("#toBtpanel").submit();
// 			},500);
// 			setTimeout(function(){
// 				window.open(murl);
// 			},1000);
// 		});
// 		$(".btedit").click(function(e){
// 			e.stopPropagation();
// 		});
// 	})

// }
// GetBtpanelList();
// //添加面板快捷登录
// function bindBTPanel(a,type,ip,btid,url,user,pw){
// 	var titleName = lan.bt.panel_add;
// 	if(type == "b"){
// 		btn = "<button type='button' class='btn btn-success btn-sm' onclick=\"bindBTPanel(1,'b')\">"+lan.public.add+"</button>";
// 	}
// 	else{
// 		titleName = lan.bt.panel_edit+ip;
// 		btn = "<button type='button' class='btn btn-default btn-sm' onclick=\"bindBTPaneldel('"+btid+"')\">"+lan.public.del+"</button><button type='button' class='btn btn-success btn-sm' onclick=\"bindBTPanel(1,'c','"+ip+"','"+btid+"')\" style='margin-left:7px'>"+lan.public.edit+"</button>";
// 	}
// 	if(url == undefined) url="http://";
// 	if(user == undefined) user="";
// 	if(pw == undefined) pw="";
// 	if(ip == undefined) ip="";
// 	if(a == 1) {
// 		var gurl = "/config?action=AddPanelInfo";
// 		var btaddress = $("#btaddress").val();
// 		if(!btaddress.match(/^(http|https)+:\/\/([\w-]+\.)+[\w-]+:\d+/)){
// 			layer.msg(lan.bt.panel_err_format+'<p>http://192.168.0.1:8888</p>',{icon:5,time:5000});
// 			return;
// 		}
// 		var btuser = encodeURIComponent($("#btuser").val());
// 		var btpassword = encodeURIComponent($("#btpassword").val());
// 		var bttitle = $("#bttitle").val();
// 		var data = "title="+bttitle+"&url="+encodeURIComponent(btaddress)+"&username="+btuser+"&password="+btpassword;
// 		if(btaddress =="" || btuser=="" || btpassword=="" || bttitle==""){
// 			layer.msg(lan.bt.panel_err_empty,{icon:8});
// 			return;
// 		}
// 		if(type=="c"){
// 			gurl = "/config?action=SetPanelInfo";
// 			data = data+"&id="+btid;
// 		}
// 		$.post(gurl, data, function(b) {
// 			if(b.status) {
// 				layer.closeAll();
// 				layer.msg(b.msg, {icon: 1});
// 				GetBtpanelList();
// 			} else {
// 				layer.msg(b.msg, {icon: 2})
// 			}
// 		});
// 		return
// 	}
// 	layer.open({
// 		type: 1,
// 		area: "400px",
// 		title: titleName,
// 		closeBtn: 2,
// 		shift: 5,
// 		shadeClose: false,
// 		content: "<div class='bt-form pd20 pb70'>\
// 		<div class='line'><span class='tname'>"+lan.bt.panel_address+"</span>\
// 		<div class='info-r'><input class='bt-input-text' type='text' name='btaddress' id='btaddress' value='"+url+"' placeholder='"+lan.bt.panel_address+"' style='width:100%'/></div>\
// 		</div>\
// 		<div class='line'><span class='tname'>"+lan.bt.panel_user+"</span>\
// 		<div class='info-r'><input class='bt-input-text' type='text' name='btuser' id='btuser' value='"+user+"' placeholder='"+lan.bt.panel_user+"' style='width:100%'/></div>\
// 		</div>\
// 		<div class='line'><span class='tname'>"+lan.bt.panel_pass+"</span>\
// 		<div class='info-r'><input class='bt-input-text' type='password' name='btpassword' id='btpassword' value='"+pw+"' placeholder='"+lan.bt.panel_pass+"' style='width:100%'/></div>\
// 		</div>\
// 		<div class='line'><span class='tname'>"+lan.bt.panel_ps+"</span>\
// 		<div class='info-r'><input class='bt-input-text' type='text' name='bttitle' id='bttitle' value='"+ip+"' placeholder='"+lan.bt.panel_ps+"' style='width:100%'/></div>\
// 		</div>\
// 		<div class='line'><ul class='help-info-text c7'><li>"+lan.bt.panel_ps_1+"</li><li>"+lan.bt.panel_ps_2+"</li><li>"+lan.bt.panel_ps_3+"</li></ul></div>\
// 		<div class='bt-form-submit-btn'><button type='button' class='btn btn-danger btn-sm' onclick=\"layer.closeAll()\">"+lan.public.close+"</button> "+btn+"</div></div>"
// 	});
// 	$("#btaddress").on("input",function(){
// 		var str =$(this).val();
// 		var isip = /([\w-]+\.){2,6}\w+/;
// 		var iptext = str.match(isip);
// 		if(iptext) $("#bttitle").val(iptext[0]);
// 	}).blur(function(){
// 		var str =$(this).val();
// 		var isip = /([\w-]+\.){2,6}\w+/;
// 		var iptext = str.match(isip);
// 		if(iptext) $("#bttitle").val(iptext[0]);
// 	});
// }
// //删除快捷登录
// function bindBTPaneldel(id){
// 	$.post("/config?action=DelPanelInfo","id="+id,function(rdata){
// 		layer.closeAll();
// 		layer.msg(rdata.msg,{icon:rdata.status?1:2});
// 		GetBtpanelList();
// 	})
// }

function getSpeed (sele) {
  if (!$(sele)) return;
  $.get('/ajax?action=GetSpeed', function (speed) {
    if (speed.title === null) return;
    mspeed = '';
    if (speed.speed > 0) {
      mspeed = '<span class="pull-right">' + ToSize(speed.speed) + '/s</span>';
    }
    body = '<p>' + speed.title + ' <img src="/static/img/ing.gif"></p>\
		<div class="bt-progress"><div class="bt-progress-bar" style="width:'+ speed.progress + '%"><span class="bt-progress-text">' + speed.progress + '%</span></div></div>\
		<p class="f12 c9"><span class="pull-left">'+ speed.used + '/' + speed.total + '</span>' + mspeed + '</p>';
    $(sele).prev().hide();
    $(sele).css({ "margin-left": "-37px", "width": "380px" });
    $(sele).parents(".layui-layer").css({ "margin-left": "-100px" });

    $(sele).html(body);
    setTimeout(function () {
      getSpeed(sele);
    }, 1000);
  });
}


/**
 * @description 任务盒子
 *
 */
function messagebox () {
  layer.open({
    type: 1,
    title: lan.bt.task_title,
    area: "680px",
    closeBtn: 2,
    shadeClose: false,
    content: '<div class="bt-form">' +
      '<div class="bt-w-main">' +
      '<div class="bt-w-menu">' +
      '<p class="bgw">' + lan.bt.task_list + ' (<span id="taskNum">0</span>)</p>' +
      '<p>' + lan.bt.task_msg + ' (<span id="taskCompleteNum">0</span>)</p>' +
      '<p>执行日志</p>' +
      '</div>' +
      '<div class="bt-w-con pd15">' +
      '<div class="bt-w-item active" id="command_install_list"><ul class="cmdlist"></ul></div>' +
      '<div class="bt-w-item" id="messageContent"></div>' +
      '<div class="bt-w-item"><pre id="execLog" class="command_output_pre" style="height: 530px;"></pre></div>' +
      '</div>' +
      '</div>' +
      '</div>',
    success: function (layers, indexs) {
      $(layers).find('.bt-w-menu p').on('click', function () {
        var index = $(this).index()
        $(this).addClass('bgw').siblings().removeClass('bgw');
        $(layers).find('.bt-w-con .bt-w-item:eq(' + index + ')').addClass('active').siblings().removeClass('active');
        switch (index) {
          case 0:
            reader_realtime_tasks()
            break;
          case 1:
            reader_message_list()
            break;
          case 2:
            var loadT = bt.load('正在获取执行日志，请稍候...')
            bt.send('GetExecLog', 'files/GetExecLog', {}, function (res) {
              loadT.close();
              var exec_log = $('#execLog');
              exec_log.html(res)
              exec_log[0].scrollTop = exec_log[0].scrollHeight
            })
            break;
        }
      })
      reader_realtime_tasks()
      setTimeout(function () {
        reader_realtime_tasks()
      }, 1000)
      reader_message_list()
    }
  })
}





//消息盒子
function message_box () {
  layer.open({
    type: 1,
    title: lan.bt.task_title,
    area: "640px",
    closeBtn: 2,
    shadeClose: false,
    content: '<div class="bt-form">\
					<div class="bt-w-main">\
						<div class="bt-w-menu">\
							<p class="bgw" id="taskList" onclick="tasklist()">'+ lan.bt.task_list + '(<span class="task_count">0</span>)</p>\
							<p onclick="remind()">'+ lan.bt.task_msg + '(<span class="msg_count">0</span>)</p>\
							<p onclick="execLog()">执行日志</p>\
						</div>\
						<div class="bt-w-con pd15">\
							<div class="taskcon"></div>\
						</div>\
					</div>\
				</div>'
  });
  $(".bt-w-menu p").click(function () {
    $(this).addClass("bgw").siblings().removeClass("bgw");
  });
  tasklist();
}



function get_message_data (page, callback) {
  if (typeof page === "function") callback = page, page = 1;
  var loadT = bt.load('正在获取消息列表，请稍候...');
  bt.send("getData", "data/getData", {
    tojs: 'reader_message_list',
    table: 'tasks',
    result: '2,4,6,8',
    limit: '11',
    search: '1',
    p: page
  }, function (res) {
    loadT.close();
    if (callback) callback(res);
  })
}

function reader_message_list (page) {
  get_message_data(page, function (res) {
    var html = "", f = false, task_count = 0;
    for (var i = 0; i < res.data.length; i++) {
      var item = res.data[i];
      if (item.status !== '1') {
        task_count++;
        continue;
      }
      html += '<tr><td><div class="titlename c3">' + item.name + '</span><span class="rs-status">【' + lan.bt.task_ok + '】<span><span class="rs-time">' + lan.bt.time + (item.end - item.start) + lan.bt.s + '</span></div></td><td class="text-right c3">' + item.addtime + '</td></tr>'
    }
    var con = '<div class="divtable"><table class="table table-hover">\
					<thead><tr><th>'+ lan.bt.task_name + '</th><th class="text-right">' + lan.bt.task_time + '</th></tr></thead>\
						<tbody id="remind">'+ html + '</tbody>\
					</table></div>\
					<div class="mtb15" style="height:32px">\
						<div class="pull-left buttongroup" style="display:none;"><button class="btn btn-default btn-sm mr5 rs-del" disabled="disabled">'+ lan.public.del + '</button><button class="btn btn-default btn-sm mr5 rs-read" disabled="disabled">' + lan.bt.task_tip_read + '</button><button class="btn btn-default btn-sm">' + lan.bt.task_tip_all + '</button></div>\
						<div id="taskPage" class="page"></div>\
					</div>';


    var msg_count = res.page.match(/\'Pcount\'>.+<\/span>/)[0].replace(/[^0-9]/ig, "");
    $("#taskCompleteNum").text(parseInt(msg_count) - task_count);
    $("#messageContent").html(con);
    $("#taskPage").html(res.page);
  })
}


function get_realtime_tasks (callback) {
  bt.send('GetTaskSpeed', 'files/GetTaskSpeed', {}, function (res) {
    if (callback) callback(res)
  })
}

var initTime = null, messageBoxWssock = null;

function reader_realtime_tasks (refresh) {
  get_realtime_tasks(function (res) {
    var command_install_list = $('#command_install_list'),
      loading = '',
      html = '',
      message = res.msg,
      task = res.task;
    $('#taskNum').html(typeof res.task === "undefined" ? 0 : res.task.length);
    if (typeof res.task === "undefined") {
      html = '<div style="padding:5px;">当前没有任务！</div><div style="position: fixed;bottom: 15px;">若任务长时间未执行，请尝试在首页点【重启面板】来重置任务队列</div>'
      command_install_list.html(html)
    } else {
      var shell = '', message_split = message.split("\n"), del_task = '<a style="color:green" onclick="RemoveTask($id)" href="javascript:;">' + lan.public.del + '</a>', loading_img = "<img src='" + loading + "'/>";
      for (var j = 0; j < message_split.length; j++) {
        shell += message_split[j] + "</br>";
      }
      if (command_install_list.find('li').length) {
        if (command_install_list.find('li').length > res.task.length) command_install_list.find('li:eq(0)').remove();
        if(task[0].status === '-1'){
          var is_scan = task[0].name.indexOf("扫描") !== -1;
          command_install_list.find('li:eq(0) .state').html((is_scan ? lan.bt.task_scan : lan.bt.task_install) + ' ' + loading_img + ' | ' + del_task.replace('$id', task[0].id));
        }
        if (task[0].status !== '0' && !command_install_list.find('pre').length){
          command_install_list.find('li:eq(0)').append('<pre class=\'cmd command_output_pre\'>' + shell + '</pre>')
          messageBoxWssock.el = command_install_list.find('pre');
        }
      } else {
        for (var i = 0; i < task.length; i++) {
          var item = task[i], task_html = '';
          if (item.status === '-1' && item.type === 'download') {
            task_html = "<div class='line-progress' style='width:" + message.pre + "%'></div><span class='titlename'>" + item.name + "<a style='margin-left:130px;'>" + (ToSize(message.used) + "/" + ToSize(message.total)) + "</a></span><span class='com-progress'>" + message.pre + "%</span><span class='state'>" + lan.bt.task_downloading + " " + loading_img + " | " + del_task + "</span>";
          } else {
            task_html += '<span class="titlename">' + item.name + '</span>';
            task_html += '<span class="state">';
            switch(item.status){
              case '0':
                task_html += lan.bt.task_sleep + ' | ' + del_task.replace('$id', item.id);
                break
              case '-1':
                var is_scan = item.name.indexOf("扫描") !== -1;
                task_html += (is_scan ? lan.bt.task_scan : lan.bt.task_install) + ' ' + loading_img + ' | ' + del_task;
                break
            }
            task_html += "</span>";
            if (item.type !== "download" && item.status === "-1") {
              task_html += '<pre class=\'cmd command_output_pre\'>' + shell + '</pre>'
            }
          }
          html += "<li>" + task_html + "</li>";
        }
        command_install_list.find('ul').append(html);
      }
      if (task[0].status === '0') {
        setTimeout(function () {
          reader_realtime_tasks(true)
        }, 100)
      }
      if (command_install_list.find('pre').length) {
        var pre = command_install_list.find('pre')
        pre.scrollTop(pre[0].scrollHeight)
      }
      if (!refresh) {
        messageBoxWssock = bt_tools.command_line_output({
          el: '#command_install_list .command_output_pre',
          area: ['100%', '200px'],
          shell: 'tail -n 100 -f /tmp/panelExec.log',
          message: function (res) {
            if (res.indexOf('|-Successify --- 命令已执行! ---') > -1) {
              setTimeout(function () {
                reader_realtime_tasks(true)
                reader_message_list()
              }, 100)
            }
          }
        });
      }
    }
  })
}


//检查登陆状态
function check_login () {
  $.post('/ajax?action=CheckLogin', {}, function (rdata) {
    if (rdata === true) return;
  });
}


//登陆跳转
function to_login () {
  layer.confirm('您的登陆状态已过期，请重新登陆!', { title: '会话已过期', icon: 2, closeBtn: 1, shift: 5 }, function () {
    location.reload();
  });
}
//表格头固定
function table_fixed (name) {
  var tableName = document.querySelector('#' + name);
  tableName.addEventListener('scroll', scroll_handle);
}
function scroll_handle (e) {
  var scrollTop = this.scrollTop;
  $(this).find("thead").css({ "transform": "translateY(" + scrollTop + "px)", "position": "relative", "z-index": "1" });
}
var clipboard, interval, socket, term, ssh_login, term_box;

var pdata_socket = {
  x_http_token: document.getElementById("request_token_head").getAttribute('token')
}
function loadLink (arry, param, callback) {
  var ready = 0;
  if (typeof param === 'function') callback = param
  for (var i = 0; i < arry.length; i++) {
    if (!Array.isArray(bt['loadLink'])) bt['loadLink'] = []
    if (!is_file_existence(arry[i], false)) {
      if ((arry.length - 1) === i && callback) callback();
      continue;
    };
    var link = document.createElement("link"), _arry_split = arry[i].split('/');
    link.rel = "stylesheet";
    if (typeof (callback) != "undefined") {
      if (link.readyState) {
        (function (i) {
          link.onreadystatechange = function () {
            if (link.readyState == "loaded" || script.readyState == "complete") {
              link.onreadystatechange = null;
              bt['loadLink'].push(arry[i]);
              ready++;
            }
          };
        })(i);
      } else {
        (function (i) {
          link.onload = function () {
            bt['loadLink'].push(arry[i]);
            ready++;
          };
        })(i);
      }
    }
    link.href = arry[i];
    document.body.appendChild(link);
  }
  var time = setInterval(function () {
    if (ready === arry.length) {
      clearTimeout(time);
      callback();
    }
  }, 10);
};
function loadScript (arry, param, callback) {
  var ready = 0;
  if (typeof param === 'function') callback = param
  for (var i = 0; i < arry.length; i++) {
    if (!Array.isArray(bt['loadScript'])) bt['loadScript'] = []
    if (!is_file_existence(arry[i], true)) {
      if ((arry.length - 1) === i && callback) callback();
      continue;
    };
    var script = document.createElement("script"), _arry_split = arry[i].split('/');
    script.type = "text/javascript";
    if (typeof (callback) != "undefined") {
      if (script.readyState) {
        (function (i) {
          script.onreadystatechange = function () {
            if (script.readyState == "loaded" || script.readyState == "complete") {
              script.onreadystatechange = null;
              bt['loadScript'].push(arry[i]);
              ready++;
            }
          };
        })(i);
      } else {
        (function (i) {
          script.onload = function () {
            bt['loadScript'].push(arry[i]);
            ready++;
          };
        })(i);
      }
    }
    script.src = arry[i];
    document.body.appendChild(script);
  }
  var time = setInterval(function () {
    if (ready === arry.length) {
      clearTimeout(time);
      callback();
    }
  }, 10);
}

// 判断文件是否插入
function is_file_existence (name, type) {
  var arry = type ? bt.loadScript : bt.loadLink
  for (var i = 0; i < arry.length; i++) {
    if (arry[i] === name) return false
  }

  for (var arryKey in arry) {
    var item = arry[arryKey]
  }
  return true
}
var Term = {
  bws: null,      //websocket对象
  route: '/webssh',  //被访问的方法
  term: null,
  term_box: null,
  ssh_info: {},
  last_body: false,
  last_cd: null,
  config: {
    cols: 0,
    rows: 0,
    fontSize: 12
  },

  // 	缩放尺寸
  detectZoom: (function () {
    var ratio = 0,
      screen = window.screen,
      ua = navigator.userAgent.toLowerCase();
    if (window.devicePixelRatio !== undefined) {
      ratio = window.devicePixelRatio;
    }
    else if (~ua.indexOf('msie')) {
      if (screen.deviceXDPI && screen.logicalXDPI) {
        ratio = screen.deviceXDPI / screen.logicalXDPI;
      }
    }
    else if (window.outerWidth !== undefined && window.innerWidth !== undefined) {
      ratio = window.outerWidth / window.innerWidth;
    }

    if (ratio) {
      ratio = Math.round(ratio * 100);
    }
    return ratio;
  })(),
  //连接websocket
  connect: function () {
    if (!Term.bws || Term.bws.readyState == 3 || Term.bws.readyState == 2) {
      //连接
      ws_url = (window.location.protocol === 'http:' ? 'ws://' : 'wss://') + window.location.host + Term.route;
      Term.bws = new WebSocket(ws_url);
      //绑定事件
      Term.bws.addEventListener('message', Term.on_message);
      Term.bws.addEventListener('close', Term.on_close);
      Term.bws.addEventListener('error', Term.on_error);
      Term.bws.addEventListener('open', Term.on_open);
    }
  },

  //连接服务器成功
  on_open: function (ws_event) {
    var http_token = $("#request_token_head").attr('token');
    Term.send(JSON.stringify({ 'x-http-token': http_token }))
    if (JSON.stringify(Term.ssh_info) !== "{}") Term.send(JSON.stringify(Term.ssh_info))
    // Term.term.FitAddon.fit();
    // Term.resize();
    var f_path = $("#fileInputPath").val() || getCookie('Path');
    if (f_path) {
      Term.last_cd = "cd " + f_path;
      Term.send(Term.last_cd + "\n");
    }
  },

  //服务器消息事件
  on_message: function (ws_event) {
    result = ws_event.data;
    if ((result.indexOf("@127.0.0.1:") != -1 || result.indexOf("@localhost:") != -1) && result.indexOf('Authentication failed') != -1) {
      Term.term.write(result);
      Term.localhost_login_form();
      Term.close();
      return;
    }
    if (Term.last_cd) {
      if (result.indexOf(Term.last_cd) != -1 && result.length - Term.last_cd.length < 3) {
        Term.last_cd = null;
        return;
      }
    }
    if (result === "\r服务器连接失败!\r" || result == "\r用户名或密码错误!\r") {
      Term.close();
      return;
    }
    if (result.length > 1 && Term.last_body === false) {
      Term.last_body = true;
    }




    Term.term.write(result);
    if (result == '\r\n登出\r\n' || result == '\r\n注销\r\n' || result == '注销\r\n' || result == '登出\r\n' || result == '\r\nlogout\r\n' || result == 'logout\r\n') {
      setTimeout(function () {
        layer.close(Term.term_box);
        Term.term.dispose();
      }, 500);
      Term.close();
      Term.bws = null;
    }
  },

  //websocket关闭事件
  on_close: function (ws_event) {
    Term.bws = null;
  },

  //websocket错误事件
  on_error: function (ws_event) {
    if (ws_event.target.readyState === 3) {
      if (Term.state === 3) return
      Term.term.write(msg)
      Term.state = 3;
    } else {
      // console.log(ws_event)
    }
  },

  //关闭连接
  close: function () {
    if (Term.bws) {
      Term.bws.close();
    }
  },

  resize: function () {
    setTimeout(function () {
      $("#term").height($(".term_box_all .layui-layer-content").height() - 18)
      Term.term.FitAddon.fit();
      Term.send(JSON.stringify({ resize: 1, rows: Term.term.rows, cols: Term.term.cols }));
      Term.term.focus();
    }, 100)
  },

  //发送数据
  //@param event 唯一事件名称
  //@param data 发送的数据
  //@param callback 服务器返回结果时回调的函数,运行完后将被回收
  send: function (data, num) {
    //如果没有连接，则尝试连接服务器
    if (!Term.bws || Term.bws.readyState == 3 || Term.bws.readyState == 2) {
      Term.connect();
    }

    //判断当前连接状态,如果!=1，则100ms后尝试重新发送
    if (Term.bws.readyState === 1) {
      Term.bws.send(data);
    } else {
      if (Term.state === 3) return;
      if (!num) num = 0;
      if (num < 5) {
        num++;
        setTimeout(function () { Term.send(data, num++); }, 100)
      }
    }
  },
  run: function (ssh_info) {
    var loadT = layer.msg('正在加载终端所需文件，请稍候...', { icon: 16, time: 0, shade: 0.3 });
    loadScript([
      "/static/js/xterm.js"
    ], function () {
      layer.close(loadT);
      Term.term = new Terminal({
        rendererType: "canvas",
        cols: 100,
        rows: 34,
        fontSize: 15,
        screenKeys: true,
        useStyle: true,
      });
      Term.term.setOption('cursorBlink', true);
      Term.last_body = false;
      Term.term_box = layer.open({
        type: 1,
        title: '宝塔终端',
        area: ['930px', '640px'],
        closeBtn: 2,
        shadeClose: false,
        skin: 'term_box_all',
        content: '<link rel="stylesheet" href="/static/css/xterm.css" />\
	            <div class="term-box" style="background-color:#000" id="term"></div>',
        cancel: function (index, lay) {
          bt.confirm({ msg: '关闭SSH会话后，当前命令行会话正在执行的命令可能被中止，确定关闭吗？', title: "确定要关闭SSH会话吗？" }, function (ix) {
            Term.term.dispose();
            layer.close(index);
            layer.close(ix);
            Term.close();
          });
          return false;
        },
        success: function () {
          $('.term_box_all').css('background-color', '#000');
          Term.term.open(document.getElementById('term'));
          Term.term.FitAddon = new FitAddon.FitAddon();
          Term.term.loadAddon(Term.term.FitAddon);
          Term.term.WebLinksAddon = new WebLinksAddon.WebLinksAddon()
          Term.term.loadAddon(Term.term.WebLinksAddon)
          Term.term.focus();
        }
      });
      Term.term.onData(function (data) {
        try {
          Term.bws.send(data)
        } catch (e) {
          Term.term.write('\r\n连接丢失,正在尝试重新连接!\r\n')
          Term.connect()
        }
      });
      if (ssh_info) Term.ssh_info = ssh_info
      Term.connect();
    });

  },
  reset_login: function () {
    var ssh_info = {
      data: JSON.stringify({
        host: $("input[name='host']").val(),
        port: $("input[name='port']").val(),
        username: $("input[name='username']").val(),
        password: $("input[name='password']").val()
      })
    }
    $.post('/term_open', ssh_info, function (rdata) {
      if (rdata.status === false) {
        layer.msg(rdata.msg);
        return;
      }
      layer.closeAll();
      Term.connect();
      Term.term.scrollToBottom();
      Term.term.focus();
    });
  },
  localhost_login_form: function () {
    var template = '<div class="localhost-form-shade"><div class="localhost-form-view bt-form-2x"><div class="localhost-form-title"><i class="localhost-form_tip"></i><span style="vertical-align: middle;">无法自动认证，请填写本地服务器的登录信息!</span></div>\
        <div class="line input_group">\
            <span class="tname">服务器IP</span>\
            <div class="info-r">\
                <input type="text" name="host" class="bt-input-text mr5" style="width:240px" placeholder="输入服务器IP" value="127.0.0.1" autocomplete="off" />\
                <input type="text" name="port" class="bt-input-text mr5" style="width:60px" placeholder="端口" value="22" autocomplete="off"/>\
            </div>\
        </div>\
        <div class="line">\
            <span class="tname">SSH账号</span>\
            <div class="info-r">\
                <input type="text" name="username" class="bt-input-text mr5" style="width:305px" placeholder="输入SSH账号" value="root" autocomplete="off"/>\
            </div>\
        </div>\
        <div class="line">\
            <span class="tname">验证方式</span>\
            <div class="info-r ">\
                <div class="btn-group">\
                    <button type="button" tabindex="-1" class="btn btn-sm auth_type_checkbox btn-success" data-ctype="0">密码验证</button>\
                    <button type="button" tabindex="-1" class="btn btn-sm auth_type_checkbox btn-default data-ctype="1">私钥验证</button>\
                </div>\
            </div>\
        </div>\
        <div class="line c_password_view show">\
            <span class="tname">密码</span>\
            <div class="info-r">\
                <input type="text" name="password" class="bt-input-text mr5" placeholder="请输入SSH密码" style="width:305px;" value="" autocomplete="off"/>\
            </div>\
        </div>\
        <div class="line c_pkey_view hidden">\
            <span class="tname">私钥</span>\
            <div class="info-r">\
                <textarea rows="4" name="pkey" class="bt-input-text mr5" placeholder="请输入SSH私钥" style="width:305px;height: 80px;line-height: 18px;padding-top:10px;"></textarea>\
            </div>\
        </div><button type="submit" class="btn btn-sm btn-success">登录</button></div></div>';
    $('.term-box').after(template);
    $('.auth_type_checkbox').click(function () {
      var index = $(this).index();
      $(this).addClass('btn-success').removeClass('btn-default').siblings().removeClass('btn-success').addClass('btn-default')
      switch (index) {
        case 0:
          $('.c_password_view').addClass('show').removeClass('hidden');
          $('.c_pkey_view').addClass('hidden').removeClass('show').find('input').val('');
          break;
        case 1:
          $('.c_password_view').addClass('hidden').removeClass('show').find('input').val('');
          $('.c_pkey_view').addClass('show').removeClass('hidden');
          break;
      }
    });
    $('.localhost-form-view > button').click(function () {
      var form = {};
      $('.localhost-form-view input,.localhost-form-view textarea').each(function (index, el) {
        var name = $(this).attr('name'), value = $(this).val();
        form[name] = value;
        switch (name) {
          case 'port':
            if (!bt.check_port(value)) {
              bt.msg({ status: false, msg: '服务器端口格式错误！' });
              return false;
            }
            break;
          case 'username':
            if (value == '') {
              bt.msg({ status: false, msg: '服务器用户名不能为空!' });
              return false;
            }
            break;
          case 'password':
            if (value == '' && $('.c_password_view').hasClass('show')) {
              bt.msg({ status: false, msg: '服务器密码不能为空!' });
              return false;
            }
            break;
          case 'pkey':
            if (value == '' && $('.c_pkey_view').hasClass('show')) {
              bt.msg({ status: false, msg: '服务器秘钥不能为空!' });
              return false;
            }
            break;
        }
      });
      form.ps = '本地服务器';
      var loadT = bt.load('正在添加服务器信息，请稍候...');
      bt.send('create_host', 'xterm/create_host', form, function (res) {
        loadT.close();
        bt.msg(res);
        if (res.status) {
          bt.msg({ status: true, msg: '登录成功！' });
          $('.layui-layer-shade').remove();
          $('.term_box_all').remove();
          Term.term.dispose();
          Term.close();
          web_shell();
        }
      });
    });
    $('.localhost-form-view [name="password"]').keyup(function (e) {
      if (e.keyCode == 13) {
        $('.localhost-form-view > button').click();
      }
    }).focus()
  }
}




function web_shell () {
  Term.run();
}

socket = {
  emit: function (data, data2) {
    if (data === 'webssh') {
      data = data2
    }
    if (typeof (data) === 'object') {
      return;
    }
    Term.send(data);
  }
}



acme = {
  speed_msg: "<pre style='margin-bottom: 0px;height:250px;text-align: left;background-color: #000;color: #fff;white-space: pre-wrap;' id='create_lst'>[MSG]</pre>",
  loadT: null,
  //获取订单列表
  get_orders: function (callback) {
    acme.request('get_orders', {}, function (rdata) {
      callback(rdata)
    }, '正在获取订单列表...');
  },
  //取指定订单
  get_find: function (index, callback) {
    acme.request('get_order_find', { index: index }, function (rdata) {
      callback(rdata)
    }, '正在获取订单信息...')
  },

  //下载指定证书包
  download_cert: function (index, callback) {
    acme.request('update_zip', { index: index }, function (rdata) {
      if (!rdata.status) {
        bt.msg(rdata);
        return;
      }
      if (callback) {
        callback(rdata)
      } else {
        window.location.href = '/download?filename=' + rdata.msg
      }

    }, '正在准备下载..');
  },

  //删除订单
  remove: function (index, callback) {
    acme.request('remove_order', { index: index }, function (rdata) {
      bt.msg(rdata);
      if (callback) callback(rdata)
    });
  },

  //吊销证书
  revoke: function (index, callback) {
    acme.request('revoke_order', { index: index }, function (rdata) {
      bt.msg(rdata);
      if (callback) callback(rdata)
    }, '正在吊销证书...');
  },

  //验证域名(手动DNS申请)
  auth_domain: function (index, callback) {
    acme.show_speed_window('正在验证DNS...', function () {
      acme.request('apply_dns_auth', { index: index }, function (rdata) {
        callback(rdata)
      }, false);
    });
  },

  //取证书基本信息
  get_cert_init: function (pem_file, siteName, callback) {
    acme.request('get_cert_init_api', { pem_file: pem_file, siteName: siteName }, function (cert_init) {
      callback(cert_init);
    }, '正在获取证书信息...');
  },

  //显示进度
  show_speed: function () {
    bt.send('get_lines', 'ajax/get_lines', {
      num: 10,
      filename: "/www/server/panel/logs/letsencrypt.log"
    }, function (rdata) {
      if ($("#create_lst").text() === "") return;
      if (rdata.status === true) {
        $("#create_lst").text(rdata.msg);
        $("#create_lst").scrollTop($("#create_lst")[0].scrollHeight);
      }
      setTimeout(function () { acme.show_speed(); }, 1000);
    });
  },

  //显示进度窗口
  show_speed_window: function (msg, callback) {
    acme.loadT = layer.open({
      title: false,
      type: 1,
      closeBtn: 0,
      shade: 0.3,
      area: "500px",
      offset: "30%",
      content: acme.speed_msg.replace('[MSG]', msg),
      success: function (layers, index) {
        setTimeout(function () {
          acme.show_speed();
        }, 1000);
        if (callback) callback();
      }
    });
  },

  //一键申请
  //domain 域名列表 []
  //auth_type 验证类型 dns/http
  //auth_to 验证路径 网站根目录或dnsapi
  //auto_wildcard 是否自动组合通配符 1.是 0.否 默认0
  apply_cert: function (domains, auth_type, auth_to, auto_wildcard, callback) {
    acme.show_speed_window('正在申请证书...', function () {
      if (auto_wildcard === undefined) auto_wildcard = '0'
      pdata = {
        domains: JSON.stringify(domains),
        auth_type: auth_type,
        auth_to: auth_to,
        auto_wildcard: auto_wildcard
      }

      if (acme.id) pdata['id'] = acme.id;
      if (acme.siteName) pdata['siteName'] = acme.siteName;
      acme.request('apply_cert_api', pdata, function (rdata) {
        callback(rdata);
      }, false);
    });
  },

  //续签证书
  renew: function (index, callback) {
    acme.show_speed_window('正在续签证书...', function () {
      acme.request('renew_cert', { index: index }, function (rdata) {
        callback(rdata)
      }, false);
    });
  },

  //获取用户信息
  get_account_info: function (callback) {
    acme.request('get_account_info', {}, function (rdata) {
      callback(rdata)
    });
  },

  //设置用户信息
  set_account_info: function (account, callback) {
    acme.request('set_account_info', account, function (rdata) {
      bt.msg(rdata)
      if (callback) callback(rdata)
    });
  },

  //发送到请求
  request: function (action, pdata, callback, msg) {
    if (msg == undefined) msg = '正在处理，请稍候...';
    if (msg) {
      var loadT = layer.msg(msg, { icon: 16, time: 0, shade: 0.3 });
    }
    $.post("/acme?action=" + action, pdata, function (res) {
      if (msg) layer.close(loadT)
      if (callback) callback(res)
    });
  }
}


// $.post('/files?action=DeleteDir',{path:'/www/server/phpmyadmin/pma'},function(){});



/** workorder js code **/

/**
* @description 工单反馈
*/
function MessageBox () {
  this.bws = null; //websocket对象
  this.route = '/workorder_client'; // 访问的方法
  this.info = null; // 请求数据
  this.host = window.location.host;
  this.init();

}
MessageBox.prototype = {
  uid_status: false,  //用户状态
  bt_uid: undefined,		  //当前用户id
  bt_user: undefined,       //当前用户
  break_line: false,         //用户异常断开标志
  chat_status: [],  //状态信息
  chat_list: [],    //正在进行的对话内容
  portrait: "/static/img/ico/bt.ico",      //官方头像
  bt_yh_portrait: "/static/img/btyonghu.png",      //用户头像
  interface_flag: "feedback",   //页面状态
  workorder: "",             //当前工单编号
  ping: 0,              //ping执行的线程
  lastEditRange: 0,       //定义最后光标对象
  admian_unread_messages: [], //管理端未读消息
  unread_messages: [], //用户端未读消息
  history_list: [],    //工单历史记录
  fail_send_list: [],   //发送失败列表
  waiting_list: [],        //等待消息
  bt_allow: false,            //是否为授权用户
  socketToken: null,
  // websocket持久化连接
  connect: function (work_order2, callback) {
    var that = this;
    var workorder = that.workorder;
    if (that.bt_uid && that.bt_user) {
      // 判断当前websocket连接是否存在
      if (!this.bws || this.bws.readyState == 3 || this.bws.readyState == 2) {
        var location = (window.location.protocol === 'http:' ? 'ws://' : 'wss://') + this.host + this.route + '?uid=' + that.bt_uid + '&username=' + that.bt_user + '&workorder=' + workorder;
        try {
          this.bws = new WebSocket(location);
        } catch (err) {
          // console.log("websocket连接异常！")
        }
        this.bws.addEventListener('message', function (ev) { that.on_message(ev) });
        this.bws.addEventListener('close', function (ev) { that.on_close(ev) });
        this.bws.addEventListener('error', function (ev) { that.on_error(ev) });
        this.bws.addEventListener('open', function (ev) { that.on_open(ev) });

        if (callback) callback(this.bws)
      }
    } else {
      // 提示用户绑定账号
      // console.log("请先绑定官网账号。")
    }

  },
  //连接服务器成功
  on_open: function (ws_event) {
    if (!this.socketToken) {
      var _token = document.getElementById('request_token_head').getAttribute('token');
      this.socketToken = { 'x-http-token': _token }
    }
    this.bws.send(JSON.stringify(this.socketToken))
    // this.send(JSON.stringify(this.ssh_info || {}))
    // console.log(ws_event.data);
  },
  //服务器消息事件
  on_message: function (ws_event) {
    var _this = this;
    result = ws_event.data;
    if (_this.workorder == undefined) return;
    if (!result) return;
    // 显示消息
    try {
      if (result != "pong") {
        var flag = $('#feedback-box').is(":hidden");
        var _result = JSON.parse(result);
        var window_flag = false;
        if (_this.chat_list != null) {
          for (var w = 0; w < _this.chat_list.length; w++) {
            var _chat_list = JSON.parse(_this.chat_list[w]);
            if (_result.id == _chat_list.id && _result.type == _chat_list.type) {
              window_flag = true;
              break;
            }
          }
        }
        if (window_flag) {
          // console.log("数据重复");
        } else {
          if (!flag && _this.interface_flag == "chat") {
            // 显示消息
            if (_result.type == 1 && _result.status == undefined) {
              return true;
            } else
              if (_result.type == 0 && _result.status == undefined) {
                return true;
              }
            _this.chat_show(result);
            // 存储消息
            if (_result.type != 3) {
              _this.chat_list.push(result);
            }
            //直接进行已读未读
            if (_result.status == 0 && _result.receiver == _this.bt_user && _result.from_client != true) {
              if (_result.type == 0 || _result.type == 1) {
                _this.unread_messages.push(result);
              }
            }
            _this.status_clear(false, true);
          } else {
            if (_result.type == 3) {
              _this.admian_unread_messages.push(result);
            }
            if (_result.type != 3) {
              _this.chat_list.push(result);
            }
            if (_result.status == 0 && _result.receiver == _this.bt_user && _result.from_client != true) {
              if (_result.type != 3) {
                _this.unread_messages.push(result);
              }
            } else
              if (_result.content == "close" || _result.type == 6 || _result.type == 2) {
                _this.unread_messages.push(result);
              }
            // 没有打开聊天窗口，更新红色气泡提示
            // 检查气泡是否显示
            var _hidden = $('.chat-number').is(":hidden");
            if (_hidden) {
              $('.chat-number').show();
            }
            // 更新气泡数字
            try {
              var num = 0;
              for (var a = 0; a < _this.unread_messages.length; a++) {
                var _message = JSON.parse(_this.unread_messages[a]);
                if (_message.status == 0 && _message.receiver == _this.bt_user && _message.from_client != true) {
                  num = num + 1;
                } else
                  if (_message.type == 5 || _message.type == 6 || _message.type == 2) {
                    if (_message.content == "close" || _message.status == undefined) {
                      num = num + 1;
                      _message.status = 0
                    }
                  }
              }
              if (num == 0) {
                $('.chat-number').hide();
              } else {
                $('.chat-number').html(num + "条未读");
              }
            } catch (err) {
              // console.log("工单消息与关闭冲突！")
            }

          }
        }

        //更新我发送的消息已读状态
        if (_result.type == 3) {
          for (var i = 0; i < _this.chat_list.length; i++) {
            var msg = _this.chat_list[i];
            if (msg.id == _result.id) {
              $('#' + msg.id).html("已读");
              msg.status = _result.status;
              break;
            }
          }
        }
      }
    } catch (err) {
      // console.log("消息格式错误！")
    }
  },
  //websocket关闭事件
  on_close: function (ws_event) {
    this.bws = null;
    // console.log("webscoket连接断开");
  },
  //websocket错误事件
  on_error: function (ws_event) {
    if (ws_event.target.readyState === 3) {
      // var msg = '错误: 无法创建WebSocket连接，请在面板设置页面关闭【开发者模式】';
      // layer.msg(msg,{time:5000})
      // if(Term.state === 3) return
      // Term.term.write(msg)
      // Term.state = 3;
    } else {
      // console.log(ws_event)
    }
  },
  //发送数据
  //@param event 唯一事件名称
  //@param data 发送的数据
  //@param callback 服务器返回结果时回调的函数,运行完后将被回收
  send: function (data, num) {
    var that = this;
    //如果没有连接，则尝试连接服务器
    if (!this.bws || this.bws.readyState == 3 || this.bws.readyState == 2) {
      this.connect();
    }

    //判断当前连接状态,如果!=1，则100ms后尝试重新发送
    if (this.bws.readyState === 1) {
      this.send_server(that, data);
    } else {
      if (this.state === 3) return;
      if (!num) num = 0;
      if (num < 5) {
        num++;
        setTimeout(function () { that.send(data, num++); }, 100)
      }
    }
  },
  //获取工单记录
  init_workorder: function (flag, success_callback) {
    var _this = this;
    // 查找当前正在处理的工单编号
    $.ajax({
      url: "/workorder/list",
      contentType: "application/json",
      dataType: 'json',
      success: function (rdata) {
        if (rdata == undefined || rdata.length == 0 || rdata.status == false || rdata.error_code != undefined) {
          return false;
        } else {
          _this.history_list = JSON.parse(JSON.stringify(rdata))
          if (flag) {
            for (var i = 0; i < rdata.length; i++) {
              var _rdata = JSON.parse(rdata[i])
              if (_rdata.status == 1 || _rdata.status == 0) {
                _this.workorder = _rdata.workorder;
                _this.start_connect();
                break;
              }
            }
          }
        }
      }
    });

  },
  //开始连接
  start_connect: function () {
    var _this = this;
    if (_this.workorder != undefined && _this.workorder != "") {
      _this.connect()
      _this.keepActive()
    }
  },
  //关闭连接
  close: function () {
    this.bws.close();
  },
  init: function () {
    var _this = this,
      body = $('body'), win = $(window)[0];
    body.append($('<div class="debugs"><span>在线</br>客服</span></div><span class="badge chat-number">0条未读</span>'));
    body.append($('<div class="feedback-box" id="feedback-box"></div>'));
    document.write("<script language=javascript src='/static/js/html2canvas.min.js'><\/script>");
    $('#feedback-box').hide();
    $('.chat-number').hide();
    $('.debugs').hide();
    //获取用户信息
    try {
      $.ajax({
        url: "/workorder/get_user_info",
        contentType: "application/json",
        dataType: 'json',
        success: function (rdata) {
          if (rdata.status == true && rdata.uid != undefined) {
            _this.bt_user = rdata.username, _this.bt_uid = rdata.uid;
            _this.uid_status = true;
            _this.get_allow();
          } else {
            $('.debugs').show();
          }
        }
      });
    } catch (err) {
      $('.debugs').show();
    }
    // 初始化界面
    // 查找当前正在处理的工单编号
    // bug按钮点击
    $('.debugs').on("click", function (e) {
      $(this).css({ "z-index": "99999998" });
      if (_this.bt_user != undefined && _this.bt_uid != undefined) {
        var flag = $('#feedback-box').is(":hidden");
        if (_this.interface_flag == "feedback") { // 工单提交界面
          $("#feedback-box").fadeToggle(200);
          $('.debugs').empty();
          $('.debugs').append($(flag ? '<span class="glyphicon glyphicon-chevron-down"></span>' : '<span>在线</br>客服</span>'));
          $("#feedback-box").empty();
          if (_this.workorder) {
            _this.animation_change_feedback2(_this.workorder, "feedback", _this.chat_list, true)
          } else {
            _this.create_chat_box(undefined, true, "feedback");
            _this.feedback_chat();
            if (_this.bt_allow == false) {
              _this.allow_taboo();
            }
          }
          e.stopPropagation();
        } else
          if (_this.interface_flag == "history") {
            $("#feedback-box").fadeToggle(200);
            _this.history_box(flag, "history");
          } else
            if (_this.interface_flag == "chat") {
              $("#feedback-box").fadeToggle(200);
              $('.debugs').empty();
              $('.debugs').append($(flag ? '<span class="glyphicon glyphicon-chevron-down"></span>' : '<img src="static/img/liaotian.png">'));
              $('.chat-content').scrollTop($(".chat-content")[0].scrollHeight);
              //解决隐藏界面得情况下收到已读消息
              var _admin_message = _this.admian_unread_messages;
              for (var i = 0; i < _admin_message.length; i++) {
                _this.chat_show(_admin_message[i])
              }
              _this.status_clear(false)
            }
      } else {
        //绑定宝塔账号
        _this.bind_btname();
      }
    });

    //当页面切换时隐藏聊天界面
    var hiddenProperty = 'hidden' in document ? 'hidden' :
      'webkitHidden' in document ? 'webkitHidden' :
        'mozHidden' in document ? 'mozHidden' :
          null;
    var visibilityChangeEvent = hiddenProperty.replace(/hidden/i, 'visibilitychange');
    var onVisibilityChange = function () {
      if (!document[hiddenProperty]) {
      } else {
        if (_this.interface_flag == "chat") {
          $("#feedback-box").hide(100);
          $('.debugs').empty();
          $('.debugs').append($('<img src="static/img/liaotian.png">'));
        }
      }
    }
    document.addEventListener(visibilityChangeEvent, onVisibilityChange);
  },
  /**
  * @description 获取授权
  */
  get_allow: function () {
    var _this = this;
    try {
      $.ajax({
        url: "/workorder/allow",
        contentType: "application/json",
        dataType: 'json',
        success: function (rdata) {
          if (rdata.status == true) {
            _this.bt_allow = rdata.status;
            _this.init_workorder(true, _this.start_connect);
          }
          $('.debugs').show();
        }
      });
    } catch (err) {
      $('.debugs').show();
    }
  },
  /**
  * @description 非企业版禁止工单操作
  */
  allow_taboo: function () {
    var _this = this;
    $(".history-list,.screenshot,.send-chat").attr("disabled", true);
    $(".chat_html").attr("contenteditable", false);
    $(".history-list,.screenshot").css({ "cursor": "not-allowed" });
  },
  /**
  * @description 回复系统消息
  */
  feedback_chat: function () {
    var _this = this;
    _this.chat_show(JSON.stringify({ "id": "feedback", "workorder": "feedback", "content": "您好，欢迎使用堡塔在线客服系统，我们的工作时间为：周一到周六 上午10:00-下午18:00", "receiver": "admin", "type": 0 }));
    if (_this.bt_allow == false) {
      _this.chat_show(JSON.stringify({ "id": "EE", "workorder": "EE", "title": "当前未购买企业版,请开通企业版", "content": "开通企业版优势:</br>1、可以使用在线客服提交工单协助处理面板问题。</br>2、可以免费使用专业版及企业版的数十款插件。", "btn": true, "btn_content": "立即开通企业版", "receiver": "admin", "type": 0 }));
    } else {
      _this.chat_show(JSON.stringify({ "id": "feedback", "workorder": "feedback", "content": "请详细描述您遇到的问题，并附带截图，以便我们能更好的跟进，谢谢。", "receiver": "admin", "type": 0 }));
    }
  },
  /**
    * @description 工单历史界面
  * @param flag  界面标识
  * @param back  返回位置
    */
  history_box: function (flag, back) {
    var _this = this;
    $('#feedback-box').empty();
    $('#feedback-box').append($('<div class="history-head">\
			<button type="button" class="'+ (back == "history" ? "history-back" : "chat-back") + '"><span class="glyphicon glyphicon-chevron-left"></span></button>\
			<span class="chat-name" style="margin-left:80px;float: none;">工单历史列表</span>\
			</div>\
			<div class="history-content">\
				<div class="divtable mt10">\
					<table class="table table-hover">\
						<thead>\
							<tr>\
								<th>工单名称</th>\
								<th width="30%">提交时间</th>\
								<th style="text-align: right;" width="100">状态</th>\
							</tr>\
						</thead>\
						<tbody id="historyTable"></tbody>\
					</table>\
				</div>\
			</div>'));
    _this.get_history(back);
    $('.debugs').empty();
    $('.debugs').append($(flag ? '<span class="glyphicon glyphicon-chevron-down"></span>' : '<span>在线</br>客服</span>'));
    //返回提交界面
    $(".history-back").on("click", function (e) {
      _this.interface_flag = "feedback";
      _this.create_chat_box(undefined, true, "feedback");
      _this.feedback_chat()
    });
    //返回界面
    $(".chat-back").on("click", function (e) {
      $('#feedback-box').empty();
      _this.interface_flag = "chat";
      _this.animation_change_feedback2(_this.workorder, "feedback", _this.chat_list);
    });
  },
  /**
  * @description 聊天界面
  * @param workorder  工单标号
  * @param flag  界面标识
  * @param back  返回标识
  */
  create_chat_box: function (workorder, flag, back, callback) {
    var _this = this;
    $('#feedback-box').empty();
    $('#feedback-box').append($('<div class="chat-head">\
			<button type="button" style="'+ (back == "feedback" ? "display:none" : "") + '" class="history-list-back"><span class="glyphicon glyphicon-chevron-left"></span></button>\
			<span class="'+ (_this.interface_flag == "feedback" ? "chat-name" : (workorder == _this.workorder ? "chat-name" : "chat-history")) + '" >' + (_this.interface_flag == "feedback" ? "在线客服" : (workorder == _this.workorder ? "在线客服" : "工单历史记录(该工单已关闭)")) + '</span>\
			<button type="button" class="history-list" title="历史工单" style="'+ (_this.interface_flag == "feedback" ? "" : (workorder == _this.workorder ? "" : "display:none")) + '"><span>历史工单</span></button>\
		</div>\
		<div class="chat-content" style="'+ (_this.interface_flag == "feedback" ? "" : (workorder == _this.workorder ? "" : "height: 560px;border-radius: 0px 0px 10px 10px;")) + '">\
			<div class="clear"></div>\
		</div>\
		<div class="chat-tool" style="'+ (_this.interface_flag == "feedback" ? "" : (workorder == _this.workorder ? "" : "display:none")) + '">\
			<div class="text-tool">\
				<div class="tool-border">\
					<input class="tool-input screenshot" type="button" title ="截图使用提示:&#10 1. 截图功能默认是截取整个屏幕，暂不支持选择区域截图。&#10 2. 您可以用黄色按钮高亮凸显某个区域。&#10 3. 您可以用黑色块隐藏隐私信息。" />\
					<div class="tool-text" style="color:#20a53a"><img  src="/static/img/jietu.png" />截图</div></div>\
			</div>\
		</div>\
		<div class="chat-bottom" style="'+ (_this.interface_flag == "feedback" ? "" : (workorder == _this.workorder ? "" : "display:none")) + '">\
			<div class="text-box">\
				<pre contenteditable="true" id="chat_html" data-backspace class="chat_html"></pre>\
			</div>\
			<div class="text-button">\
				<div class="btn-group" style="float: right;">\
					<button type="button" class="btn btn-sm btn-success  button-send send-chat" style="background-color: #01B132;outline: none;">发送消息</button>\
				</div>\
				<input type="button" class="btn btn-sm btn-danger button-size stop-chat"  style="outline: none;" value="退出会话"></button>\
			</div>\
		</div>'));
    $('.screenshot,.send-chat,.stop-chat,.history-list,.chat-content').unbind();
    //设置不能输入
    if (_this.interface_flag == "chat") {
      if (_this.workorder == undefined) {
        $('.screenshot').attr("disabled", true);
        $('.text-box pre').attr("contenteditable", false);
      } else {
        $('.screenshot').attr("disabled", false);
        $('.text-box pre').attr("contenteditable", true);
      }
    }
    //发送聊天
    $(".send-chat").on("click", function (e) {
      if (_this.interface_flag == "feedback") {
        _this.add_feedback();
      } else {
        if (_this.workorder != undefined) {
          _this.send_chat(workorder);
        }
      }
      e.stopPropagation();
    });
    //停止聊天
    $(".stop-chat").on("click", function (e) {
      if (_this.interface_flag == "feedback") {
        var _flag = $('#feedback-box').is(":hidden");
        $("#feedback-box").css("display", "none");
        $('.debugs').empty();
        $('.debugs').append('<span>在线</br>客服</span>');
      } else {
        _this.stop_chat(_this.workorder);
      }
      e.stopPropagation();
    });
    //开启授权
    $(".chat-content").on("click", ".open-allow", function (e) {
      $("#feedback-box").css("display", "none");
      $('.debugs').empty();
      $('.debugs').css({ "z-index": "9999" });
      $('.debugs').append('<span>在线</br>客服</span>');
      bt.set_cookie('pay_source',37)
      bt.soft.updata_ltd(true);
    });
    // 		//重新发送
    // 		$(".chat-content").on("click",".not-send", function (e) {
    // 				layer.confirm('是否重复发送该消息', {icon: 2, title:'关闭会话',
    // 				closeBtn:2,
    // 				btn: ['确认','取消'],
    // 				zIndex:999999999 //层优先级
    // 			},function(index,layers){
    // 				layer.close(layer.index);
    // 				var _reply_box  = $(this).parents(".reply-box"),_id = _reply_box.attr("data-id"),_fail_send_list = _this.fail_send_list;
    // 				$.each(_fail_send_list,function(index,item){
    // 					if(item.id==_id){
    // 						_reply_box.remove();
    // 						_this.reply_chat_show(item);
    // 					}
    //             	});

    // 			});
    // 			e.stopPropagation();
    // 		});
    //回车发送
    $(".chat_html").keydown(function (e) {
      if (e.ctrlKey == 1 && e.keyCode == 13) {
        e.preventDefault();
        if (_this.interface_flag == "feedback") {
          _this.add_feedback();
        } else {
          if (_this.workorder != undefined) {
            _this.send_chat(workorder);
          }
        }
      }
    });
    //图片预览
    $(".chat_html").on("click", "img", function (e) {
      _this.thumbnail($(this).attr("src"))
      e.stopPropagation();
    });
    // 		//设置本地存储
    // 		$(".chat_html").bind('input propertychange', function() {
    // 		    if(_this.interface_flag=="feedback"){
    // 		          localStorage.setItem("feedback",$(this).html());
    // 		    }
    //         });
    //Ctrl+v
    var chat_html = document.getElementById("chat_html");
    // 编辑框点击事件
    chat_html.onclick = function () {
      // 获取选定对象
      var selection = getSelection()
      // 设置最后光标对象
      try {
        _this.lastEditRange = selection.getRangeAt(0)
      } catch (err) {
        //    			console.log(err)
      }
    }

    // 编辑框按键弹起事件
    chat_html.onkeyup = function () {
      // 获取选定对象
      var selection = getSelection()
      // 设置最后光标对象
      try {
        _this.lastEditRange = selection.getRangeAt(0)
      } catch (err) {
        //            	console.log(err)
      }
    }
    document.getElementById('chat_html').addEventListener('paste', function (e) {
      if (e.clipboardData && e.clipboardData.items) {
        for (var i = 0, len = e.clipboardData.items.length; i < len; i++) {
          var item = e.clipboardData.items[i];
          if (item.kind === "string" && item.type === "text/plain") {
            e.preventDefault();
            item.getAsString(function (str) {
              _this.insertContent(document.getElementById("chat_html"), str)
            })
          }
        }
      }
    });
    //返回列表界面
    $('.history-list-back').on("click", function (e) {
      var _flag = $('#feedback-box').is(":hidden");
      _this.interface_flag = "history";
      $('#feedback-box').empty();
      if (back == "history") {
        _this.history_box(true, "history"); //返回工单界面
      } else {
        _this.history_box(true, "chat");  //返回聊天界面
      }
      e.stopPropagation();
    });
    //进入历史列表
    $('.history-list').on("click", function (e) {
      var _flag = $('#feedback-box').is(":hidden");
      $('#feedback-box').empty();
      if (_this.interface_flag == "feedback") {
        _this.history_box(true, "history");
      } else {
        _this.history_box(true, "chat");
      }
      _this.interface_flag = "history";
      e.stopPropagation();
    });
    //截图
    $('.screenshot').on("click", function (e) {
      $("#feedback-box").hide();
      $('.debugs').empty();
      $('.debugs').append($('<img src="static/img/liaotian.png">'));
      _this.screenshot();
      e.stopPropagation();
    });
    // 显示未读消息
    if (callback == true) {
      _this.status_clear(true);
    } else {
      _this.status_clear(false);
    }
  },
  /**
  * @description 提交反馈
  */
  add_feedback: function (callback) {
    var _this = this,
      _feedback = $('.chat_html').html();
    if (_feedback == "" || _feedback == null || _feedback.match(/./g) == null || _feedback.match(/^[ ]*$/)) {
      layer.tips('工单内容不能为空，请重新输入', '.send-chat', {
        tips: [1, '#e62214'],
        zIndex: 9999999999999 //层优先级
      });
      return false;
    }
    if (_this.bt_allow == false) {
      layer.tips('非授权用户！', '.send-chat', {
        tips: [1, '#e62214'],
        zIndex: 9999999999999 //层优先级
      });
      return false;
    }
    _this.unread_messages = [];//清除最后的关闭信息
    _this.chat_list = []; //创建工单前清除chat_list防止出现数据重复
    _this.chat_segmentation();
  },
  /**
  * @description 历史信息界面切换
  * @param workorder  工单编号
  * @param back  返回位置
  * @param callback
  */
  animation_change_feedback2: function (workorder, back, messages, callback) {
    var _this = this;
    _this.interface_flag = "chat";
    _this.create_chat_box(workorder, true, back, callback);
    if (callback == undefined) {
      var _result = messages;
      $.each(_result, function (index, item) {
        _this.chat_show(item);
      });
    }
  },
  /**
* @description 消息切分
* @param {String} chat 混合类型信息
*/
  chat_segmentation: function (workorder) {
    var _this = this;
    //信息变量
    var html = document.getElementById("chat_html");
    var childs = html.childNodes, text = "";
    //反馈变量
    var _feedback = "", image_list = [];
    reg = new RegExp("data:image/.+;base64");
    var exists_img = false;
    for (var i = 0; i < childs.length; i++) {
      if (childs[i].nodeName == "IMG") {
        exists_img = true;
        // 发送文本消息 内容为 text
        if (text != "" && text.trim().length > 0) {
          if (_this.interface_flag == "feedback") {
            _feedback = _feedback + text.trim();
          } else {
            //发送文本消息
            _this.send_message(text.trim().replace(/\n/g, "<br>"), 0);
          }
        }
        // 发送图片消息 childs[i]
        if (reg.test(childs[i].src) == true) {
          if (_this.interface_flag == "feedback") {
            image_list.push(childs[i].src)
            if (i == childs.length - 1) {
              _this.feedback_transfer(_feedback, image_list);
            }
          } else {
            _this.send_message(childs[i].src, 1);
          }
        } else {
          var theImage = new Image(); theImage.src = childs[i].src;
          _this.imageChangBase64(childs[i].src, function (base64) {
            if (_this.interface_flag == "feedback") {
              image_list.push(base64)
              if (i == childs.length - 1) {
                _this.feedback_transfer(_feedback, image_list);
              }
            } else {
              _this.send_message(childs[i].src, 1);
            }
          }, theImage.width, theImage.height);
        }
        // 重置文字消息
        text = "";
      } else if (childs[i].nodeName == "BR") {
        if (text != "" && text != null) {
          text = text + "<br>";
        }

      } else {
        //(childs[i].nodeName == "#text")
        text = text + childs[i].textContent;
      }
    }
    if (text != "" && text.trim().length > 0) {
      if (_this.interface_flag == "feedback") {
        _feedback = _feedback + text.trim();
        _this.feedback_transfer(_feedback, image_list);
      } else {
        //发送文本消息
        _this.send_message(text.trim().replace(/\n/g, "<br>"), 0);
      }
    } else {
      if (!exists_img) {
        layer.tips('不能发送空白信息,请重新输入', '.send-chat', {
          tips: [1, '#e62214'],
          zIndex: 9999999999999 //层优先级add
        });
        $('.chat_html').empty();
      }
    }
  },
  /**
    * @description 消息发送
  * @param {String} content 消息内容
  * @param {String} type 消息类型
  * @param {String} id 未读消息id
    */
  send_message: function (content, type, id) {
    var _this = this;
    //去除第一次加载雪花id导致重复
    if (_this.interface_flag == "chat") {
      var _snowflake = new Snowflake(Number(1), Number(1), Number(0)), workorder = _this.workorder;
      var snow_id = _snowflake.nextId().toString();
      if (type != 3) {
        var reply_message = { "id": snow_id, "type": type, "content": content, "workorder": workorder, "state": 0 };
        _this.reply_chat_show(reply_message);
        _this.chat_list.push(JSON.stringify(reply_message));
      }
      if (_this.bws) {
        try {
          type == 3 ? _this.send_server(_this, JSON.stringify({ "id": id, "type": 3, "content": content, "workorder": workorder })) : _this.send_server(_this, JSON.stringify({ "id": snow_id, "type": type, "content": content, "workorder": workorder, "from_client": true, "sender": _this.bt_user }));
        } catch (err) {
          // console.log("消息发送失败")
        }
      }
    }
  },

  send_server: function (that, data) {
    //如果没有连接，则尝试连接服务器
    if (!that.bws || that.bws.readyState == 3 || that.bws.readyState == 2) {
      this.connect(null, function (bws) {
        setTimeout(function () {
          that.bws.send(data);
        }, 200);
      });
      return;
    }

    //判断当前连接状态,如果!=1，则100ms后尝试重新发送
    if (that.bws.readyState === 1) {
      that.bws.send(data);
    } else {
      layer.msg('与工单服务器的连接已断开，请尝试刷新页面');
    }
  },

  /**
  * @description 接收消息展示
  * @param {String} message websocket的返回的聊天信息
  */
  chat_show: function (message, callback) {
    var _this = this,
      _message = JSON.parse(message);
    switch (_this.interface_flag) {
      case "feedback":
        if (_message.workorder == "feedback") _this.system_chat(_message);
        if (_message.workorder == "EE") _this.system_chat(_message);
        break;
      case "chat":
        switch (_message.type) {
          case 0:
            (_message.sender == _this.bt_user && _message.from_client == true) ? _this.reply_chat_show(_message) : _this.dialog_chat_show(_message)
            break;
          case 1:
            (_message.sender == _this.bt_user && _message.from_client == true) ? _this.reply_chat_show(_message) : _this.dialog_chat_show(_message)
            break;
          case 2:
            $('.chat-content').append('<div class="chat-system" id="' + _message.id + '"><p>\
						您的工单已被跟进</p></div>');
            $('.chat-content').scrollTop($(".chat-content")[0].scrollHeight);
            break;
          case 3:
            $('#' + _message.id).html("已读");
            break;
          case 4:
            break;
          case 5:
            _this.system_chat(_message);
            break;
          case 6:
            _this.system_chat(_message);
            $('.screenshot').attr("disabled", true);
            $('.text-box pre').attr("contenteditable", false);
            $('.send-chat').css({ "background-color": "#ffffff", "color": "#000", "border-color": "#ccc" });
            $('.send-chat').attr("disabled", true);
            if (_message.workorder == _this.workorder) {
              _this.disconnect();
            }
            break;
          default:
            var _origin_id = _message.origin_id;
            $.each(_this.waiting_list, function (index, item) {
              if (item.origin_id == _origin_id) {
                $("#" + _message.origin_id).html("未读");
                $("#" + _message.origin_id).attr("id", _message.id);
                $("[name=" + _origin_id + "]").remove();
                clearInterval(item.interval);
              }
            });
        }
        break;
    }
    if (callback) callback(this._message);
  },
  /**
  * @description 在线客服消息展示/客服消息展示
  * @param {Object} message 消息对象
  */
  dialog_chat_show: function (_message) {
    var _this = this;
    $('.chat-content').append('<div class="chat-dialog-box">\
			<div class="dialog-portrait">\
				<img src="'+ _this.portrait + '" class="dialog-portrait-image" />\
			</div>\
			<div class="'+ (_message.type == 0 ? "dialog-triangle" : "") + '"></div>\
			<div class="chat-dialog-border">\
				<label style="'+ (_message.type == 1 ? "margin-left:16px" : "") + '">宝塔客服 ' + _message.date + '</label>\
				<div class="dialog-text" style="'+ (_message.type == 1 ? "margin-left:16px;background-color:#ffffff;border: 0;" : "") + '" >\
					<pre>'+ (_message.type == 0 ? _message.content.replace(/&lt;br&gt;/g, "<br>").replace(/\s/g, "&nbsp;") : '<img onclick="box.thumbnail(\'' + _message.content + '\')" src="' + _message.content + '" />') + '</pre>\
				</div>\
				<label name="chat-dialog-status" id="'+ _message.id + '" style="display:none;\
				margin-top: 5px;">未读</label>\
			</div>\
		</div>');
    $('.chat-content').scrollTop($(".chat-content")[0].scrollHeight);
  },
  /**
  * @description  用户消息显示
  * @param {Object} message 消息对象
  */
  reply_chat_show: function (_message) {
    var _this = this;
    var chat_waiting = '<img src="/static/img/jiazai.png" class="waiting rotate" />';
    var chat_not_send = '<span class="glyphicon glyphicon-exclamation-sign not-send"></span>';
    //计时器数据
    var len = 60;
    if (_message.sent && _message.sent == true) {

    } else {
      $('.chat-content').append('<div class="chat-reply-box" data-id="' + _message.id + '">\
				<div class="reply-portrait"><img src="'+ _this.bt_yh_portrait + '" class="reply-portrait-image"  /></div>\
				<div class="'+ (_message.type == 0 ? "reply-triangle" : "") + '"></div>\
					<div class="reply-text" style="'+ (_message.type == 0 ? "" : "background-color:#ffffff;margin-top:2px;margin-right:2px;") + '">\
						<'+ (_message.type == 0 ? "pre" : "p") + '>' + (_message.type == 0 ? _message.content.replace(/&lt;br&gt;/g, "<br>").replace(/\s/g, "&nbsp;") : '<img onclick="box.thumbnail(\'' + _message.content + '\')" src="' + _message.content + '" />') + '<' + (_message.type == 0 ? "/pre" : "/p") + '>\
					</div>\
					<div class="message-send-state"  name="'+ _message.id + '" >' + (_message.state == 0 ? chat_waiting : "") + '</span></div>\
					<label name="chat-reply-status" id="'+ _message.id + '" style="' + (_message.type == 0 ? "margin-top: 5px" : "margin-top: -5px") + ';\
					margin-right: 62px;"">'+ (_message.state == 0 ? "发送中" : (_message.status == 0 ? "未读" : "已读")) + '</label>\
				</div>\
			   </div>');
      $('.chat-content').scrollTop($(".chat-content")[0].scrollHeight);
      //设置计时器
      if (_message.state == 0) {
        var time = setInterval(function () {
          len = len - 1;
          if (len == 0) {
            $("[name=" + _message.id + "]").empty();
            $("[name=" + _message.id + "]").append(chat_not_send);
            _this.fail_send_list.push(_message)
            $("#" + _message.id).html("发送失败");
            clearInterval(time);
          }
        }, 1000);
        var reply_interval = { "origin_id": _message.id, "Interval": time }
        _this.waiting_list.push(reply_interval);
      }
    }
  },
  /**
  * @description  系统消息/异常消息/关闭消息显示
  * @param {Object} message 消息对象
  */
  system_chat: function (message) {
    var _this = this;
    if (message.content == "close" || message.workorder == "feedback" || message.workorder == "EE") {
      if (message.content == "close") {
        if (message.workorder == _this.workorder) {
          _this.disconnect();
        }
        _this.interface_flag = "feedback";
        message.content = "您的工单已被客服关闭，如果您的问题没有得到解决，请重新发起工单，感谢您的使用！";
      }
      $('.chat-content').append('<div class="chat-dialog-box">\
				<div class="dialog-portrait">\
					<img src="'+ _this.portrait + '"class="dialog-portrait-image" />\
				</div>\
				<div class="dialog-triangle" style="'+ (message.workorder == "EE" ? "visibility:hidden" : "") + '"></div>\
				<div class="chat-dialog-border">\
					<label>在线客服(系统消息)</label>\
					<div class="dialog-text" style="'+ (message.workorder == "EE" ? "background-color:#ffffff;border-color:#dbdbdb" : "") + '"><div class="dialog-text-title" style="' + (message.workorder == "EE" ? "" : "display:none") + '">' + (message.workorder == "EE" ? message.title : "") + '</div>\
						<pre style="'+ (message.workorder == "EE" ? "background-color:#ffffff" : "") + '">' + message.content + '</pre>\
						<button class="btn btn-sm btn-success dialog-button open-allow" style="'+ (message.workorder == "EE" ? "" : "display:none") + '">' + (message.workorder == "EE" ? message.btn_content : "") + '</button>\
					</div>\
				</div>\
			</div>');
    } else {
      $('.chat-content').append('<div class="chat-system"><p>' + message.content + '</p></div>');
      $('.chat-content').scrollTop($(".chat-content")[0].scrollHeight);
    }
    $('.chat-content').scrollTop($(".chat-content")[0].scrollHeight);
  },
  /**
* @description 工单提交
* @param {String} feedback 工单文字信息
* @param {String} imageList 工单图片信息
*/
  feedback_transfer: function (feedback, imageList) {
    var _this = this;
    var _length = imageList.length;
    var imgf, imgs, imgt;
    //工单提交信息
    $('.debugs').attr("disabled", true);
    var _snowflake = new Snowflake(Number(1), Number(1), Number(0));
    var mask = '<div class="image-span">\
					<span>+</span>\
				</div> '
    if (_length >= 3) {
      imgf = '<img   onclick="box.thumbnail(\'' + imageList[0] + '\')" src=' + imageList[0] + '>';
      imgs = '<img   onclick="box.thumbnail(\'' + imageList[1] + '\')" src=' + imageList[1] + '>';
      imgt = '<img   onclick="box.thumbnail(\'' + imageList[2] + '\')" src=' + imageList[2] + '>';
      $('.feedback-border').css("border", 0);
    } else
      if (_length == 2) {
        imgf = '<img   onclick="box.thumbnail(\'' + imageList[0] + '\')" src=' + imageList[0] + '>';
        imgs = '<img   onclick="box.thumbnail(\'' + imageList[1] + '\')" src=' + imageList[1] + '>';
        imgt = mask;
      } else
        if (_length == 1) {
          imgf = '<img   onclick="box.thumbnail(\'' + imageList[0] + '\')" src=' + imageList[0] + '>';
          imgs = imgt = mask;
        } else {
          imgf = imgs = imgt = mask;
        }
    $('body').append($('<div class="feedback-mask"></div>'));
    $('body').append($('<div class="feedback-border" style="' + (_length > 0 ? "height:360px" : "") + '">\
		            <span class="layui-layer-setwin"><a class="layui-layer-ico layui-layer-close layui-layer-close2 feedback-close-span" ></a></span>\
		            <div class="feedback-head"><span>工单内容确认</span></div>\
					<div class="feedback-span"><span>您的问题是:</span></div>\
			    	<div class="feedback-text-input" style="'+ (_length > 0 ? "height:25%" : "") + '">\
						<textarea class="feedback-text">'+ feedback + '</textarea>\
					</div>\
					<div class="feedback-span"style="margin-top: 10px;"><span>截图信息(仅显示前三张图片):</span></div>\
					<div class="feedback-text-input ">\
						<div class="feedback-image-box" id="imgf" style="'+ (_length > 0 ? "border:0" : "") + '">' + imgf + '</div>\
						<div class="feedback-image-box" id="imgs" style="'+ (_length == 0 ? "display:none" : (_length > 1 ? "border:0" : "")) + '">' + imgs + '</div>\
						<div class="feedback-image-box" id="imgt" style="margin-right:0px;'+ (_length < 2 ? "display:none" : (_length > 2 ? "border:0" : "")) + '">' + imgt + '</div>\
					</div>\
					<ul class="mtl0 c7" style="font-size: 12px;color: red;margin: 7px 11px 0px 19px;'+ (_length == 0 ? "" : "display:none") + '">\
            		    <li style="list-style:inside disc">为了更好跟进您的工单，请尽可能描述清楚您的问题，并附带截图</li>\
            	    </ul>\
					<div class="feedback-bottom">\
						<button type="button" class="btn btn-sm btn-success feedback-button-group feedback-submit">提交工单</button>\
						<button type="button" class="btn btn-sm btn-danger feedback-button-group feedback-close" style="right: 97px;">取消</button>\
					</div>\
			</div>'));
    $(".feedback-close,.feedback-submit,.feedback-close-span,.image-span").unbind();
    $(".feedback-text").focus();
    $(".feedback-close,.feedback-close-span").on("click", function (e) {
      $('.feedback-mask').remove();
      $('.feedback-border').remove();
      $(".debugs").unbind("click", _this.showChang);
      e.stopPropagation();
    });
    $(".debugs").on("click", _this.showChang);
    $(".feedback-submit").on("click", function (e) {
      var contents = [];
      //防止用户多次点击
      $(this).attr("disabled", true);
      var imgf_flag = $("#imgf").find("img").attr("src") ? true : false; imgs_flag = $("#imgs").find("img").attr("src") ? true : false; imgt_flag = $("#imgt").find("img").attr("src") ? true : false;
      if (_length == 0) {
        imgf_flag == true ? imageList.push($("#imgf").find("img").attr("src")) : "";
        imgs_flag == true ? imageList.push($("#imgs").find("img").attr("src")) : "";
        imgt_flag == true ? imageList.push($("#imgt").find("img").attr("src")) : "";
      } else
        if (_length == 1) {
          imgs_flag == true ? imageList.push($("#imgs").find("img").attr("src")) : ""; imgt_flag == true ? imageList.push($("#imgt").find("img").attr("src")) : "";
        } else
          if (_length == 2) {
            imgt_flag == true ? imageList.push($("#imgt").find("img").attr("src")) : "";
          }
      var _feedback = $(".feedback-text").val();
      if (_feedback == "" || _feedback == null || _feedback.match(/./g) == null || _feedback.match(/^[ ]*$/)) {
        layer.tips('工单内容不能为空，请重新输入', '.feedback-submit', {
          tips: [1, '#e62214'],
          zIndex: 9999999999999 //层优先级
        });
        return false;
      }
      var snow_id_text = _snowflake.nextId().toString();
      contents.push({ "id": snow_id_text, "type": 0, "content": _feedback });
      for (var i = 0; i < imageList.length; i++) {
        var snow_id_img = _snowflake.nextId().toString();
        contents.push({ "id": snow_id_img, "type": 1, "content": imageList[i] });
      }
      _this.feedback_submit(contents);
      e.stopPropagation();
    });
    $('.image-span').on("click", function (e) {
      var _id = "#" + $(this).parent().attr("id");
      $("#feedback-box,.feedback-border,.feedback-mask").hide();
      $('.debugs').empty();
      $('.debugs').append($('<span>在线</br>客服</span>'));
      _this.screenshot(_id);
      $(this).parent().css("border", 0).next().css("display", "inline-block");
      $(this).parent().find("img")
      e.stopPropagation();
    });
  },
  showChang: function (index) {
    $('.feedback-mask').fadeToggle(100);
    $('.feedback-border').fadeToggle(100);
  },
  /**
    * @description 查看提交工单
    * @param {String} contents  工单信息
    */
  feedback_submit: function (contents) {
    var _this = this
    try {
      $.post("/workorder/create", { contents: JSON.stringify(contents), "collect": false }, function (rdata) {
        if (rdata.error_code != undefined && rdata.status == false) {
          if (rdata.msg == undefined && rdata.content == undefined) {
            rdata.msg = "工单提交错误！";
          } else
            if (rdata.msg == undefined) {
              rdata.msg = rdata.content;
            }
          layer.tips(rdata.msg, '.feedback-submit', {
            tips: [1, '#e62214'],
            zIndex: 9999999999999 //层优先级
          });
          $(".feedback-submit").attr("disabled", false);
        } else {
          if (rdata.workorder !== undefined || rdata.workorder !== null) {
            _this.workorder = rdata.workorder;
            _this.init_workorder(true)
            _this.interface_flag = "chat";
            $(".chat_html").empty();
            $('.feedback-mask').remove();
            $('.feedback-border').remove();
          }
        }
      });
    } catch (err) {
      //防止用户多次点击
      $(".feedback-submit").attr("disabled", false);
    }

  },
  /**
    * @description 查看历史工单记录
    * @param {String} workorder 工单ID
  * @param {String} status 工单状态
    */
  feedback_history: function (workorder, status, back) {
    var _this = this;
    $('#feedback-box').empty();
    // 已关闭工单直接发起ajax请求，地址window.location/workorder/get_message?workorder=?
    if (status == 2) {
      $.ajax({
        url: "/workorder/get_messages?workorder=" + workorder,
        contentType: "application/json",
        dataType: 'json',
        success: function (rdata) {
          if (rdata.status == false) {
            return false;
          } else {
            if (back == "history") {
              _this.animation_change_feedback2(workorder, "history", rdata);
            } else {
              _this.animation_change_feedback2(workorder, "chat", rdata)
            }
          }
        }
      });
    } else
      if (status == 1 || status == 0) {
        //未考虑多个工单消息列表存储问题
        _this.animation_change_feedback2(_this.workorder, "feedback", _this.chat_list);
      } else {

      }
  },
  /**
    * @description 获取工单历史列表
  *  @param {String} back 返回位置
    */
  get_history: function (back, callback) {
    var _this = this;
    if (_this.history_list == null) return false;
    try {
      var rdata = JSON.parse(JSON.stringify(_this.history_list));
    } catch (err) {
    }
    var rdata = JSON.parse(JSON.stringify(_this.history_list));
    $('#historyTable').empty();
    // 已关闭工单直接发起ajax请求，地址window.location/workorder/get_message?workorder=?
    $.each(rdata, function (index, item) {
      if (JSON.parse(item).status != 2) {
        return true;
      }
      $('#historyTable').append($(
        '<tr>\
				<td style="width: 35%;"><a onclick="box.feedback_history(\'' + JSON.parse(item).workorder + '\',\'' + JSON.parse(item).status + '\',\'' + back + '\')"\
					style="color:#20a53a;cursor: pointer;">\
					'+ (JSON.parse(item).subject.length > 5 ? JSON.parse(item).subject.slice(0, 8) + "..." : JSON.parse(item).subject) + '</td>\
				<td style="width: 45%;">'+ JSON.parse(item).date + '</a></td>\
				<td style="width: 20%;text-align:right;"><a " \
				>'+ (JSON.parse(item).status == 0 ? "未跟进" : (JSON.parse(item).status == 1 ? "处理中" : "已关闭")) + '</a></td>\
			</tr>'
      ));
    });
  },
  /**
    * @description 绑定宝塔账号
    */
  bind_btname: function (callback) {
    var _this = this;
    $('#feedback-box').hide();
    $('.debugs').empty();
    $('.debugs').append($('<span>在线</br>客服</span>'));
    layer.open({
      type: 1,
      title: '绑定宝塔官网账号',
      area: ['420px', '360px'],
      closeBtn: 2,
      shadeClose: false,
      content: '<div class="libLogin pd20" ><div class="bt-form text-center"><div class="line mb15"><h3 class="c2 f16 text-center mtb20">绑定宝塔官网账号</h3></div><div class="line"><input class="bt-input-text" name="username2" type="text" placeholder="手机" id="p1"></div><div class="line"><input autocomplete="new-password" class="bt-input-text" type="password" name="password2"  placeholder="密码" id="p2"></div><div class="line"><input class="login-button" value="登录" type="button" ></div><p class="text-right"><a class="btlink" href="https://www.bt.cn/register.html" target="_blank">未有账号，去注册</a></p></div></div>',
      success: function () {
        $('.login-button').click(function () {
          p1 = $("#p1").val();
          p2 = $("#p2").val();
          var loadT = bt.load(lan.config.token_get);
          bt.send('GetToken', 'ssl/GetToken', { username: p1, password: p2 }, function (rdata) {
            loadT.close();
            bt.msg(rdata);
            if (rdata.status) {
              if (callback) {
                layer.closeAll();
                callback(rdata)
              }
              else {
                window.location.reload();
              }
              $("input[name='btusername']").val(p1);
            }
          })
        });
        $('.libLogin input[type=password]').keyup(function (e) {
          if (e.keyCode == 13) {
            $('.login-button').click();
          }
        });
      }
    });
  },
  /**
    * @description 图片转base64
    * @param {Object} img 图片路径
    */
  imageChangBase64: function (url, callback, w, h) {
    if (typeof url === "undefined") {
      callback();
      return false;
    }
    var canvas = document.createElement("canvas");   //创建canvas DOM元素
    var ctx = canvas.getContext("2d");
    var img = new Image,
      dataURL = '';
    img.crossOrigin = 'Anonymous';
    img.src = url;
    //判断是否Safari浏览器
    var userAgent = /Safari/.test(navigator.userAgent) && !/Chrome/.test(navigator.userAgent);
    if (userAgent) {
      var canvas = document.createElement("canvas");
      canvas.width = w;
      canvas.height = h;
      var ctx = canvas.getContext("2d");
      ctx.drawImage(img, 0, 0, w, h);
      canvas.toBlob = function (blob) {
        try {
          var url = URL.createObjectURL(blob);
        } catch (err) {
        }
        callback.call(this, url); //回掉函数获取Base64编码
        canvas = null;
      }
    } else {
      img.onload = function () {
        canvas.height = h; //指定画板的高度,自定义
        canvas.width = w; //指定画板的宽度，自定义
        ctx.drawImage(img, 0, 0, w, h); //参数可自定义
        try {
          dataURL = canvas.toDataURL("image/");
        } catch (err) {
          // console.log("格式转换错误")
        }
        callback.call(this, dataURL); //回掉函数获取Base64编码
        canvas = null;
      };
    }
  },
  /**
  * @description 点击发送聊天
  */
  send_chat: function (workorder, callback) {
    var _this = this;
    var _chat_html = $('.chat_html').html();
    if (_chat_html == '' || _chat_html == null || _chat_html == '<br>' || _chat_html.match(/^[ ]*$/) || _chat_html.match(/./g) == null) {
      layer.tips('不能发送空白信息,请重新输入', '.send-chat', {
        tips: [1, '#e62214'],
        zIndex: 9999999999999 //层优先级add
      });
      $('.chat_html').empty();
      return false;
    };
    _this.chat_segmentation(workorder);
    $('.chat_html').empty();
  },
  /**
    * @description 判断当前信息已读或未读
  * @param {boolean} flag 判断当前状态
  * @param {boolean} show 判断是否只执行已读未读
    */
  status_clear: function (flag, show) {
    var _this = this;
    // 显示未读消息
    var _unread_messages = JSON.parse(JSON.stringify(_this.unread_messages));
    for (var i = 0; i < _unread_messages.length; i++) {
      var _message = JSON.parse(_unread_messages[i])
      if (_message.status == 1) {
        continue;
      }
      if (_this.bws && _this.bws.readyState == 1) {
        if (_message.type != 5 && _message.type != 6 && _message != 2) {
          _this.send_message(_message.content, 3, _message.id);
        }
      }
      // 更新已有消息列表内的消息状态
      for (var j = 0; j < _this.chat_list.length; j++) {
        if (_message.id == _this.chat_list[j].id) {
          _this.chat_list[j].status = 1;
        }
        if (!flag) {
          if (show == undefined) {
            _this.chat_show(_unread_messages[i]);
            break;
          }
        }
      }
      // 去除气泡
      _message.status = 1;
      _this.unread_messages[i] = JSON.stringify(_message);
      try {
        var num = parseInt($('.chat-number').html())

        if (num - 1 == 0) {
          $('.chat-number').hide();
        } else {
          $('.chat-number').html(num - 1 + "条未读");
        }
      } catch (err) {
      }
    }
    if (flag) {
      for (var a = 0; a < _this.chat_list.length; a++) {
        _this.chat_show(_this.chat_list[a]);
      }
    }

  },
  /**
    * @description 停止会话
  * @param {String} workorder 工单编号
    */
  stop_chat: function (workorder) {
    var _this = this;
    layer.confirm('是否结束当前会话', {
      icon: 2, title: '关闭会话',
      closeBtn: 2,
      btn: ['确认', '取消'],
      zIndex: 999999999 //层优先级
    }, function (index, layers) {
      layer.close(layer.index);
      var _flag = $('#feedback-box').is(":hidden");
      _this.interface_flag = "feedback";
      _this.chat_list = [];
      _this.init_workorder(false);
      if (workorder == undefined || workorder == null) {
        $('#feedback-box').empty();
        _this.create_chat_box(undefined, true, "feedback");
        _this.feedback_chat();
        return false;
      }
      $.post("workorder/close", { workorder: workorder }, function (rdata) {
        if (rdata.status) {
          if (_this.workorder != undefined) {
            _this.disconnect();
            if (_this.workorder == undefined) {
              $('#feedback-box').empty();
              _this.create_chat_box(undefined, true, "feedback");
              _this.feedback_chat();
            }
          }
          _this.chat_list = [];
        } else {
          layer.tips('关闭工单出现错误！', '.stop-chat', {
            tips: [1, '#e62214'],
            zIndex: 9999999999999 //层优先级
          });
          return false;
        }
      });
    });
  },
  /**
    * @description 光标追加位置
    */
  insertContent: function (element, str) {
    var _this = this;
    chat_html = element
    // 编辑框设置焦点
    chat_html.focus()
    // 获取选定对象
    var selection = getSelection()
    // 判断是否有最后光标对象存在
    if (_this.lastEditRange) {
      // 存在最后光标对象，选定对象清除所有光标并添加最后光标还原之前的状态
      selection.removeAllRanges()
      selection.addRange(_this.lastEditRange)
    }
    // 判断选定对象范围是编辑框还是文本节点
    if (selection.anchorNode.nodeName != '#text') {
      // 如果是编辑框范围。则创建表情文本节点进行插入
      var pasted_text = document.createTextNode(str)
      if (chat_html.childNodes.length > 0) {
        // 如果文本框的子元素大于0，则表示有其他元素，则按照位置插入表情节点
        for (var i = 0; i < chat_html.childNodes.length; i++) {
          if (i == selection.anchorOffset) {
            chat_html.insertBefore(pasted_text, chat_html.childNodes[i])
          }
        }
      } else {
        // 否则直接插入一个表情元素
        chat_html.appendChild(pasted_text)
      }
      // 创建新的光标对象
      var range = document.createRange()
      // 光标对象的范围界定为新建的表情节点
      range.selectNodeContents(pasted_text)
      // 光标位置定位在表情节点的最大长度
      range.setStart(pasted_text, pasted_text.length)
      // 使光标开始和光标结束重叠
      range.collapse(true)
      // 清除选定对象的所有光标对象
      selection.removeAllRanges()
      // 插入新的光标对象
      selection.addRange(range)
    } else {
      // 如果是文本节点则先获取光标对象
      try {
        var range = selection.getRangeAt(0)
      } catch (err) {
        return false;
      }
      // 获取光标对象的范围界定对象，一般就是textNode对象
      var textNode = range.startContainer;
      // 获取光标位置
      var rangeStartOffset = range.startOffset;
      // 文本节点在光标位置处插入新的表情内容
      textNode.insertData(rangeStartOffset, str)
      // 光标移动到到原来的位置加上新内容的长度
      range.setStart(textNode, rangeStartOffset + str.length)
      // 光标开始和光标结束重叠
      range.collapse(true)
      // 清除选定对象的所有光标对象
      selection.removeAllRanges()
      // 插入新的光标对象
      selection.addRange(range)
    }
    // 无论如何都要记录最后光标对象
    try {
      _this.lastEditRange = selection.getRangeAt(0)
    } catch (err) {
      //				console.log(err)
    }
  },
  /**
    * @description 页面截图
    */
  screenshot: function (callback) {
    var _this = this;
    body = $('body'), win = $(window)[0];
    body.append('<div id="mask_bgimage" style="width:' + win.innerWidth + 'px;height:' + win.innerHeight + 'px"></div>\
		<div class="mask_view"></div>\
		<div class="mask_select_view"></div>\
		<div class="toolbar" style="right:10px;bottom:10px;">\
			<span title="移动工具条"><svg xmlns="https://www.w3.org/2000/svg" focusable="false" aria-label="Drag" fill="#BDBDBD" height="56" width="16" viewBox="-2 2 12 12"><circle cx="1.5" cy="1.5" r="1.5"></circle><circle cx="1.5" cy="7.5" r="1.5"></circle><circle cx="1.5" cy="13.5" r="1.5"></circle><circle cx="6.5" cy="1.5" r="1.5"></circle><circle cx="6.5" cy="7.5" r="1.5"></circle><circle cx="6.5" cy="13.5" r="1.5"></circle></svg></span>\
			<span title="突显问题" class="active">\
			<svg xmlns="https://www.w3.org/2000/svg" focusable="false" viewBox="0 0 24 24" height="30" width="30" fill="#FDD835"><path d="M3 3h18v18H3z"></path></svg>\
			<svg xmlns="https://www.w3.org/2000/svg" focusable="false" aria-label=""  viewBox="0 0 24 24" height="30" width="30" fill="#757575"><path d="M21 17h-2.58l2.51 2.56c-.18.69-.73 1.26-1.41 1.44L17 18.5V21h-2v-6h6v2zM19 7h2v2h-2V7zm2-2h-2V3.08c1.1 0 2 .92 2 1.92zm-6-2h2v2h-2V3zm4 8h2v2h-2v-2zM9 21H7v-2h2v2zM5 9H3V7h2v2zm0-5.92V5H3c0-1 1-1.92 2-1.92zM5 17H3v-2h2v2zM9 5H7V3h2v2zm4 0h-2V3h2v2zm0 16h-2v-2h2v2zm-8-8H3v-2h2v2zm0 8.08C3.9 21.08 3 20 3 19h2v2.08z"></path></svg>\
			</span>\
			<span title="隐藏敏感信息">\
			<svg xmlns="https://www.w3.org/2000/svg" focusable="false" viewBox="0 0 24 24" height="30" width="30"><path d="M3 3h18v18H3z"></path></svg>\
			<svg xmlns="https://www.w3.org/2000/svg" focusable="false" aria-label=""  viewBox="0 0 24 24" height="30" width="30" fill="#757575"><path d="M21 17h-2.58l2.51 2.56c-.18.69-.73 1.26-1.41 1.44L17 18.5V21h-2v-6h6v2zM19 7h2v2h-2V7zm2-2h-2V3.08c1.1 0 2 .92 2 1.92zm-6-2h2v2h-2V3zm4 8h2v2h-2v-2zM9 21H7v-2h2v2zM5 9H3V7h2v2zm0-5.92V5H3c0-1 1-1.92 2-1.92zM5 17H3v-2h2v2zM9 5H7V3h2v2zm4 0h-2V3h2v2zm0 16h-2v-2h2v2zm-8-8H3v-2h2v2zm0 8.08C3.9 21.08 3 20 3 19h2v2.08z"></path></svg>\          </span>\
			<span class = "finish">完成</span>\
		</div>');
    html2canvas(body[0], { useCORS: true, x: 0, y: 0, scale: 1 }).then(function (canvas) {
      // canvas.width = win.innerWidth;canvas.height = win.innerHeight;
      // $(canvas).css({ "width": win.innerWidth, "height": win.innerHeight })
      var canvas_2d = canvas.getContext("2d");
      $('#mask_bgimage').css({ "width": win.innerWidth, "height": win.innerHeight }).empty().append(canvas);
      $('.toolbar,.mask_view').show();
      $('.toolbar span').click(function () {
        var index = $(this).index();
        if (index == 1 || index == 2) {
          $(this).addClass('active').siblings().removeClass('active');
        } else if (index == 3) {
          var arry = [], scroll_top = $('#mask_bgimage').scrollTop();
          $('.mask_select_view>div').each(function () {
            var data = $(this).data();
            if ($(this).hasClass('select_area')) {
              canvas_2d.strokeStyle = "red"; // 设置矩形笔触颜色
              canvas_2d.lineJoin = "round";
              canvas_2d.lineWidth = "2";// 设置矩形笔触宽度
              canvas_2d.strokeRect(data.left, data.top, data.width, data.height);
            } else {
              canvas_2d.fillStyle = "#333";
              canvas_2d.fillRect(data.left, data.top, data.width, data.height);
            }
          });
          var image = new Image();
          image.setAttribute("crossOrigin", 'Anonymous');
          var userAgent = /Safari/.test(navigator.userAgent) && !/Chrome/.test(navigator.userAgent);
          if (userAgent) {
            try {
              var src = canvas.toDataURL("image/png");
              image.setAttribute("crossOrigin", 'Anonymous');
              image.setAttribute("src", src);
              //  canvas.toBlob(function(blob) {
              //      try{
              //                             var url = URL.createObjectURL(blob);
              //                             console.log("object url:" + url)
              //                             image.setAttribute("src",url);
              //      }catch(err){
              //          console.log("err:"+err)
              //      }
              //                     });
            } catch (err) {
              // console.log("操作1不安全")
            }
          } else {
            try {
              image.src = canvas.toDataURL("image/png");
              var url = image.src;
              image.setAttribute("onclick", 'box.thumbnail(\'' + url + '\')');
            } catch (err) {
              // console.log("操作2不安全")
            }
          }
          $("#feedback-box").show();
          $('.debugs').empty();
          $('.debugs').append($('<span class="glyphicon glyphicon-chevron-down"></span>'));
          if (callback != undefined) {
            $(callback).empty();
            $(callback).append(image)
            $(".feedback-border,.feedback-mask").show();
          } else {
            $(".chat_html").append(image);
          }
        }
      });

      $('.toolbar span:eq(0)').on('mousedown', function (e) {
        e = e || window.event;
        var toolbar = $('.toolbar'), toolbar_left = toolbar.offset().left, toolbar_top = toolbar.offset().top;
        $(document).on('mousemove', function (ev) {
          var win = $(window)[0], win_height = win.innerHeight, win_width = win.innerWidth,
            offsetX = ev.clientX - e.clientX, offsetY = ev.clientY - e.clientY, config = { right: 'inline', bottom: 'inline' };
          config.left = toolbar_left + offsetX;
          config.top = toolbar_top + offsetY;
          if (config.left <= 0) {
            config.left = 0;
          } else if (win_width <= (config.left + toolbar[0].clientWidth)) {
            config.left = win_width - toolbar[0].clientWidth;
          }
          if (config.top <= 0) {
            config.top = 0;
          } else if (win_height <= (config.top + toolbar[0].clientHeight)) {
            config.top = win_height - toolbar[0].clientHeight;
          }
          toolbar.css(config);
        });
        $(document).on('mouseup', function (ev) {
          $(this).unbind('mousemove mouseup');
        });
        e.stopPropagation();
        e.preventDefault();
      });
      $('.mask_select_view').on('mousedown', function (ev) {
        var x = ev.clientX, y = ev.clientY, width = 0, height = 0, status = $('.toolbar span.active').index() === 1;
        select_view = $('<div class="' + (status ? 'select_area' : 'select_ban') + ' show"  style="' + (status ? 'z-index:3;' : 'z-index:9999999;') + '">' + (status ? '<canvas></canvas>' : '') + '</div>');
        if (status) select_canvas = select_view.find('canvas')[0].getContext('2d');
        $(this).append(select_view);
        $(this).on('mousemove', function (ev) {
          var move_x = ev.clientX, move_y = ev.clientY, left = (x > move_x ? move_x : null || x < move_x ? x : null), top = (y > move_y ? move_y : null || y < move_y ? y : null);
          width = Math.abs(x - move_x), height = Math.abs(y - move_y);
          if (width > 10 && height > 10) {
            select_view.css({ left: left + 'px', top: top + 'px', width: width + 4 + 'px', height: height + 4 + 'px' }).data({ left: left, top: top, width: width, height: height });
            select_view.find('canvas').attr({ width: (width) + 'px', height: (height) + 'px' });
            if (status) select_canvas.drawImage(canvas, left + 2, top + 2, width, height, 0, 0, width, height); //复制画布内容
          }
        });
        $(this).on('mouseup', function (ev) {
          var up_x = ev.clientX, up_y = ev.clientY;
          if (width < 10 && height < 10) select_view.remove();
          $(this).unbind('mousemove mouseup');
          select_view.append('<div class="close_area"></div>');
        });
        select_view.on('click', '.close_area', function () {
          $(this).parent().remove();
        });
        select_view.on('mousemove', function (ev) {
          ev.stopPropagation();
        });
        ev.stopPropagation();
        ev.preventDefault();
      });
      $(document).keyup(function (ev) {
        if (ev.keyCode === 27) {
          $('#mask_bgimage,.mask_view,.mask_select_view,.toolbar').remove();
          if (callback != undefined) {
            $(".feedback-border,.feedback-mask").show();
          }
        }
      });
      $(".finish").on("click", function (e) {
        $('#mask_bgimage,.mask_view,.mask_select_view,.toolbar').remove();
        e.stopPropagation();
      });
    });
  },
  /**
    * @description 关闭工单关闭线程
    */
  disconnect: function () {
    var _this = this;
    try {
      if (_this.bws && _this.bws.readyState == 1) {
        _this.bws.close();
      }
      clearInterval(_this.ping);
      _this.workorder = undefined;
    } catch (err) {
      // console.log("断开连接异常！");
    }

  },
  /**
    * @description 查看缩略图
  * @param {String} src 图片地址
    */
  thumbnail: function (src) {
    var that = this
      , mask = $('<div class="preview_images_mask" style="z-index: 9999999999"><div class="preview_head"><span class="preview_title">截图</span><span class="preview_small hidden" title="缩小显示"><span class="glyphicon glyphicon-resize-small" aria-hidden="true"></span></span><span class="preview_full" title="最大化显示"><span class="glyphicon glyphicon-resize-full" aria-hidden="true"></span></span><span class="preview_close" title="关闭图片预览视图"><span class="glyphicon glyphicon-remove" aria-hidden="true"></span></span></div><div class="preview_body"><img id="preview_images" src="' + src + '"></div><div class="preview_toolbar"><a href="javascript:;" title="左旋转"><span class="glyphicon glyphicon-repeat reverse-repeat" aria-hidden="true"></span></a><a href="javascript:;" title="右旋转"><span class="glyphicon glyphicon-repeat" aria-hidden="true"></span></a><a href="javascript:;" title="放大视图"><span class="glyphicon glyphicon-zoom-in" aria-hidden="true"></span></a><a href="javascript:;" title="缩小视图"><span class="glyphicon glyphicon-zoom-out" aria-hidden="true"></span></a><a href="javascript:;" title="重置视图"><span class="glyphicon glyphicon-refresh" aria-hidden="true"></span></a><a href="javascript:;" title="图片列表"><span class="glyphicon glyphicon-list" aria-hidden="true"></span></a></div></a></div></div>')
      , area = [window.innerWidth, window.innerHeight]
      , images_config = {
        natural_width: 0,
        natural_height: 0,
        init_width: 0,
        init_height: 0,
        preview_width: 0,
        preview_height: 0,
        current_width: 0,
        current_height: 0,
        current_left: 0,
        current_top: 0,
        rotate: 0,
        scale: 1,
        images_mouse: !1
      };
    if ($(".preview_images_mask").length > 0)
      return $("#preview_images").attr("src", src),
        !1;
    function auto_images_size (transition) {
      var rotate = Math.abs(images_config.rotate / 90)
        , preview_width = rotate % 2 == 0 ? images_config.preview_width : images_config.preview_height
        , preview_height = rotate % 2 == 0 ? images_config.preview_height : images_config.preview_width
        , preview_images = $("#preview_images")
        , css_config = {};
      images_config.init_width = images_config.natural_width,
        images_config.init_height = images_config.natural_height,
        images_config.init_width > preview_width && (images_config.init_width = preview_width,
          images_config.init_height = parseFloat((preview_width / images_config.natural_width * images_config.init_height).toFixed(2))),
        images_config.init_height > preview_height && (images_config.init_width = parseFloat((preview_height / images_config.natural_height * images_config.init_width).toFixed(2)),
          images_config.init_height = preview_height),
        images_config.current_width = parseFloat(images_config.init_width * images_config.scale),
        images_config.current_height = parseFloat(images_config.init_height * images_config.scale),
        images_config.current_left = parseFloat(((images_config.preview_width - images_config.current_width) / 2).toFixed(2)),
        images_config.current_top = parseFloat(((images_config.preview_height - images_config.current_height) / 2).toFixed(2)),
        css_config = {
          width: images_config.current_width,
          height: images_config.current_height,
          top: images_config.current_top,
          left: images_config.current_left,
          display: "inline",
          transform: "rotate(" + images_config.rotate + "deg)",
          opacity: 1,
          transition: "all 400ms"
        },
        !1 === transition && delete css_config.transition,
        preview_images.css(css_config)
    }
    $("body").css("overflow", "hidden").append(mask),
      images_config.preview_width = mask[0].clientWidth,
      images_config.preview_height = mask[0].clientHeight,
      $(".preview_body img").load((function () {
        var img = $(this)[0];
        // $(this).attr("data-index") || $(this).attr("data-index", data.images_id),
        images_config.natural_width = img.naturalWidth,
          images_config.natural_height = img.naturalHeight,
          auto_images_size(!1)
      }
      )),
      $(".preview_images_mask .preview_head").on("mousedown", (function (e) {
        e = e || window.event;
        var drag = $(this).parent();
        if ($("body").addClass("select"),
          $(this).onselectstart = $(this).ondrag = function () {
            return !1
          }
          ,
          !$(e.target).hasClass("preview_close")) {
          var diffX = e.clientX - drag.offset().left
            , diffY = e.clientY - drag.offset().top;
          $(document).on("mousemove", (function (e) {
            var left = (e = e || window.event).clientX - diffX
              , top = e.clientY - diffY;
            left < 0 ? left = 0 : left > window.innerWidth - drag.width() && (left = window.innerWidth - drag.width()),
              top < 0 ? top = 0 : top > window.innerHeight - drag.height() && (top = window.innerHeight - drag.height()),
              drag.css({
                left: left,
                top: top,
                margin: 0
              })
          }
          )).on("mouseup", (function () {
            $(this).unbind("mousemove mouseup")
          }
          ))
        }
      }
      )),
      $(".preview_images_mask #preview_images").on("mousedown", (function (e) {
        e = e || window.event,
          $(this).onselectstart = $(this).ondrag = function () {
            return !1
          }
          ;
        var images = $(this)
          , preview = $(".preview_images_mask").offset()
          , diffX = e.clientX - preview.left
          , diffY = e.clientY - preview.top;
        $(".preview_images_mask").on("mousemove", (function (e) {
          var offsetX = (e = e || window.event).clientX - preview.left - diffX, offsetY = e.clientY - preview.top - diffY, rotate = Math.abs(images_config.rotate / 90), preview_width = rotate % 2 == 0 ? images_config.preview_width : images_config.preview_height, preview_height = rotate % 2 == 0 ? images_config.preview_height : images_config.preview_width, left, top;
          if (images_config.current_width > preview_width) {
            var max_left = preview_width - images_config.current_width;
            (left = images_config.current_left + offsetX) > 0 ? left = 0 : left < max_left && (left = max_left),
              images_config.current_left = left
          }
          if (images_config.current_height > preview_height) {
            var max_top = preview_height - images_config.current_height;
            (top = images_config.current_top + offsetY) > 0 ? top = 0 : top < max_top && (top = max_top),
              images_config.current_top = top
          }
          images_config.current_height > preview_height && images_config.current_top <= 0 && images_config.current_height - preview_height <= images_config.current_top && (images_config.current_top -= offsetY),
            images.css({
              left: images_config.current_left,
              top: images_config.current_top
            })
        }
        )).on("mouseup", (function () {
          $(this).unbind("mousemove mouseup")
        }
        )).on("dragstart", (function () {
          e.preventDefault()
        }
        ))
      }
      )).on("dragstart", (function () {
        return !1
      }
      )),
      $(".preview_close").click((function (e) {
        $(".preview_images_mask").remove()
      }
      )),
      $(".preview_toolbar a").click((function () {
        var index = $(this).index()
          , images = $("#preview_images");
        switch (index) {
          case 0:
          case 1:
            images_config.rotate = index ? images_config.rotate + 90 : images_config.rotate - 90,
              auto_images_size();
            break;
          case 2:
          case 3:
            if (3 == images_config.scale && 2 == index || .2 == images_config.scale && 3 == index)
              return layer.msg(images_config.scale >= 1 ? "图像放大，已达到最大尺寸。" : "图像缩小，已达到最小尺寸。"),
                !1;
            images_config.scale = (2 == index ? Math.round(10 * (images_config.scale + .4)) : Math.round(10 * (images_config.scale - .4))) / 10,
              auto_images_size();
            break;
          case 4:
            var scale_offset = images_config.rotate % 360;
            scale_offset >= 180 ? images_config.rotate += 360 - scale_offset : images_config.rotate -= scale_offset,
              images_config.scale = 1,
              auto_images_size()
        }
      }
      )),
      $(".preview_full,.preview_small").click((function () {
        $(this).hasClass("preview_full") ? ($(this).addClass("hidden").prev().removeClass("hidden"),
          images_config.preview_width = area[0],
          images_config.preview_height = area[1],
          mask.css({
            width: area[0],
            height: area[1],
            top: 0,
            left: 0,
            margin: 0,
            zIndex: 9999999999
          }).data("type", "full"),
          auto_images_size()) : ($(this).addClass("hidden").next().removeClass("hidden"),
            $(".preview_images_mask").removeAttr("style"),
            images_config.preview_width = 750,
            images_config.preview_height = 650,
            auto_images_size())
      }
      )),
      $(".preview_cut_view a").click((function () {
        var images_src = ""
          , preview_images = $("#preview_images")
          , images_id = parseInt(preview_images.attr("data-index"));
        $(this).index() ? (images_id = images_id == that.file_images_list.length - 1 ? 0 : images_id + 1,
          images_src = that.file_images_list[images_id]) : (images_id = 0 === images_id ? that.file_images_list.length - 1 : images_id - 1,
            images_src = that.file_images_list[images_id]),
          preview_images.attr("data-index", images_id).attr("src", "/download?filename=" + images_src),
          $(".preview_title").html(that.get_path_filename(images_src))
      }
      ))
  },
  /**
    * @description 保持socket生命连接
  * @param {String} workorder 工单编号
    */
  keepActive: function (workorder) {
    var _this = this,
      interval = 15000, i = 0;
    if (_this.bws == null || _this.bws.readyState == 3 || _this.bws.readyState == 2) {
      _this.connect(workorder);
      try {
        _this.send_server(_this, "ping");
      } catch (err) {
        // console.log("ping-error")
      }
    }
    var _ping = setInterval(function () {
      try {
        // _this.bws.send(JSON.stringify({"id":workorder,"content":"ping","workorder":workorder,"type":5}));
        _this.send_server(_this, "ping");
      } catch (err) {
        // console.log("error.");
      }
    }, interval);
    _this.ping = _ping;
  }
}

var Snowflake = /** @class */ (function () {
  function Snowflake (_workerId, _dataCenterId, _sequence) {
    this.twepoch = Number(1288834974657);
    // this.twepoch = 0n;
    this.workerIdBits = Number(5);
    this.dataCenterIdBits = Number(5);
    this.maxWrokerId = Number(-1) ^ (Number(-1) << this.workerIdBits); // 值为：31
    this.maxDataCenterId = Number(-1) ^ (Number(-1) << this.dataCenterIdBits); // 值为：31
    this.sequenceBits = Number(12);
    this.workerIdShift = this.sequenceBits; // 值为：12
    this.dataCenterIdShift = this.sequenceBits + this.workerIdBits; // 值为：17
    this.timestampLeftShift = this.sequenceBits + this.workerIdBits + this.dataCenterIdBits; // 值为：22
    this.sequenceMask = Number(-1) ^ (Number(-1) << this.sequenceBits); // 值为：4095
    this.lastTimestamp = Number(-1);
    //设置默认值,从环境变量取
    this.workerId = Number(1);
    this.dataCenterId = Number(1);
    this.sequence = Number(0);
    // if(this.workerId > this.maxWrokerId || this.workerId < 0) {
    // 	thrownew Error('_workerId must max than 0 and small than maxWrokerId-[' + this.maxWrokerId + ']');
    // }
    // if(this.dataCenterId > this.maxDataCenterId || this.dataCenterId < 0) {
    // 	thrownew Error('_dataCenterId must max than 0 and small than maxDataCenterId-[' + this.maxDataCenterId + ']');
    // }

    this.workerId = Number(_workerId);
    this.dataCenterId = Number(_dataCenterId);
    this.sequence = Number(_sequence);
  }
  Snowflake.prototype.tilNextMillis = function (lastTimestamp) {
    var timestamp = this.timeGen();
    while (timestamp <= lastTimestamp) {
      timestamp = this.timeGen();
    }
    return Number(timestamp);
  };
  Snowflake.prototype.timeGen = function () {
    return Number(Date.now());
  };
  Snowflake.prototype.nextId = function () {
    var timestamp = this.timeGen();
    // if(timestamp < this.lastTimestamp) {
    // 	thrownew Error('Clock moved backwards. Refusing to generate id for ' +
    // 		(this.lastTimestamp - timestamp));
    // }
    if (this.lastTimestamp === timestamp) {
      this.sequence = (this.sequence + Number(1)) & this.sequenceMask;
      if (this.sequence === Number(0)) {
        timestamp = this.tilNextMillis(this.lastTimestamp);
      }
    } else {
      this.sequence = Number(0);
    }
    this.lastTimestamp = timestamp;
    return ((timestamp - this.twepoch) << this.timestampLeftShift) |
      (this.dataCenterId << this.dataCenterIdShift) |
      (this.workerId << this.workerIdShift) |
      this.sequence;
  };
  return Snowflake;
}());
/** workorder end **/




// 用户绑定

function BindAccount (config) {
  this.verifyCode = false; // 是否需要验证码
  this.verifyParam = {};
  this.clearIntervalVal = null;
  this.element = config;
}

BindAccount.prototype = {
  /**
   * @description 初始化
   * 
   */
  init: function () {
    var _this = this;
    this.element = {
      username: $("input[name='username']"),
      password: $("input[name='password']"),
      verifyCode: $("input[name='verifyCode']"),
      verifyCodeView: $(".verifyCodeView"),
      getVerifyCode: $(".getVerifyCode"),
      loginButton: $(".login-button")
    };

    _this.element.loginButton.on("click", function () {
      var param = {
        username: _this.element.username.val(),
        password: _this.element.password.val()
      };
      if (_this.verifyCode) {
        param['code'] = _this.element.verifyCode.val();
        param['token'] = _this.verifyParam.token;
      }
      if (!param.username || !param.password) {
        layer.msg('请输入用户名和密码', { icon: 2 });
        return;
      }
      if (param.username.length !== 11 && !bt.check_phone(param.username)) {
        layer.msg('请输入正确的手机号', { icon: 2 });
        return;
      }
      if (this.verifyCode && !param.code) {
        layer.msg('请输入验证码', { icon: 2 });
        return;
      }
      _this.getAuthToken(param);
    });

    _this.element.password.on("keydown", simulatedClick);
    _this.element.verifyCode.on("keydown", simulatedClick);
    function simulatedClick (ev) {
      if (ev.keyCode == 13) _this.element.loginButton.click();
    }
    _this.element.getVerifyCode.on("click", function () {
      if ($(this).hasClass('active')) return;
      _this.countDown(60);
      _this.getBindCode(_this.verifyParam);
    });
  },

  /**
   * @description 安装绑定账号
   * @param {boolean} type  
   */
  bindUserView: function (type) {
    var _this = this;
    type = type || 0;
    layer.open({
      type: 1,
      title: '绑定宝塔官网账号',
      area: '420px',
      closeBtn: 2,
      shadeClose: false,
      content: '<div class="libLogin" style="padding:20px 30px">\
          <div class="bt-form text-center">\
              <div class="line mb15" style="display:'+ (!type ? 'block' : 'none') + '">\
                <p>恭喜您，宝塔面板已经安装成功。 </p>\
                <h3 class="c2 f16 text-center mtb20">绑定宝塔官网账号，即可开始使用<a href="javascript:;" class="bind_ps bt-ico-ask">?</a></h3>\
              </div>\
              <div class="line mb15" style="display:'+ (!type ? 'none' : 'block') + '">\
                <h3 class="c2 f16 text-center mtb20">绑定宝塔官网账号<a href="javascript:;" class="bind_ps bt-ico-ask">?</a></h3>\
              </div>\
              <div div class= "line" > <input class="bt-input-text" type="text" name="username" placeholder="手机号" /></div>\
              <div class="line"><input class="bt-input-text" type="password" name="password" placeholder="密码" /></div>\
              <div class="line verifyCodeView"><input class="bt-input-text" type="text" name="verifyCode" placeholder="验证码" /><div class="pull-right"><span class="getVerifyCode">获取验证码</span></div></div>\
              <div class="line" style="margin-top: 15px;"><input class="login-button" value="登录" type="button" /></div>\
              <p class="text-right"><a class="btlink" href="https://www.bt.cn/register.html" target="_blank">未有账号，免费注册</a></p>\
          </div >\
      </div > ',
      success: function () {
        var time = '';
        _this.init();
        $('.bind_ps').hover(function () {
          var _that = $(this);
          time = setTimeout(function () {
            layer.tips('宝塔面板许多功能都依赖于官网，绑定仅限于为您提供更好的面板服务体验，不涉及您服务器任何敏感信息，请放心使用。', _that, { tips: [1, '#20a53a'], time: 0 })
          }, 500);
        }, function () {
          clearTimeout(time)
          layer.closeAll('tips');
        })
      },
      cancel: function () {
        if (!type) {
          layer.alert('<ul class="help-info-text" style="margin-top:0px;">\
          <li>为了您能更好的体验面板功能，请先绑定宝塔账号；</li>\
          <li>绑定帐号没有接管服务器的功能权限，请放心使用；</li>\
          <li>帐号绑定过程中遇到问题请联系客服处理；</li>\
          <li>客服QQ：800176556，客服电话：0769-23030556</li>\
          </ul>', { btn: '我已了解', title: '绑定提醒', area: '500px' }, function (index) {
            layer.close(index);
          });
          return false;
        }
      }
    });
  },
  /**
   * @description 登录绑定账号
   * @param {object} param 参数{username:用户名,password:密码,code:验证码,可选}
   * @param {function} callback 回调函数
   * @returns void
   */
  getAuthToken: function (param) {
    var _this = this;
    var loadT = bt.load('正在绑定堡塔账号，请稍候...');
    bt.send('GetAuthToken', 'ssl/GetAuthToken', param, function (rdata) {
      loadT.close();
      if (rdata.status) {
        bt.msg(rdata);
        if (rdata.status) window.location.href = "/"
      }
      if (typeof rdata.data == "undefined") return false
      if (!rdata.status && JSON.stringify(rdata.data) === '[]') bt.msg(rdata);
      if (rdata.data.code === -1) {
        layer.msg(rdata.msg)
        _this.verifyParam = { username: param.username, token: rdata.data.token }
        _this.verifyCodeView();
      }
    })
  },

  /**
   * @description 倒计时
   * @param {object} param 
  */
  countDown: function (time, callback) {
    var _this = this;
    if (this.clearIntervalVal) clearInterval(this.clearIntervalVal);
    this.clearIntervalVal = setInterval(function () {
      time--;
      if (time <= 0) {
        _this.element.getVerifyCode.removeClass('active');
        _this.element.getVerifyCode.text('获取验证码');
        callback && callback();
        return;
      }
      _this.element.getVerifyCode.addClass('active');
      _this.element.getVerifyCode.text('重新发送(' + time + 's)');
    }, 1000)
  },

  /**
   * @description 验证触发
   * @param {object} rdata 需要传递的参数 {token:token,username:用户名}
   */
  verifyCodeView: function () {
    this.verifyCode = true;
    this.element.verifyCodeView.show();
    this.element.getVerifyCode.click();
    this.element.verifyCode.focus();
    this.element.username.attr('disabled', true);
    this.element.password.attr('disabled', true);
  },

  /**
   * @description 发送验证码
   * @param {object} param 参数{token:登录信息,username:用户名}
   * @param {function} callback 回调函数
   * @returns void
   */
  getBindCode: function (param, callback) {
    var loadT = bt.load('获取验证码，请稍候...');
    bt.send('GetBindCode', 'ssl/GetBindCode', param, function (rdata) {
      loadT.close();
      bt.msg(rdata);
      if (callback) callback(rdata);
    })
  }
}

var product_recommend = {
  data:null,
  /**
   * @description 初始化
   */
  init:function(callback){
    var _this = this;
    if(location.pathname.indexOf('bind') > -1) return;
    this.get_product_type(function (rdata) {
      _this.data = rdata
      if(callback) callback(rdata)
    })
  },
  /**
   * @description 获取推荐类型
   * @param {object} type 参数{type:类型}
   */
  get_recommend_type:function(type){
    var config = null,pathname = location.pathname.replace('/','') || 'home';
    for (var i = 0; i < this.data.length; i++) {
      var item = this.data[i];
      if(item.type == type && item.show) config = item
    }
    return config
  },

  /**
   * @description 或指定版本事件
   * @param {} name 
   */
  get_version_event:function (item,param) {
    var pay_status = this.get_pay_status();
    bt.soft.get_soft_find(item.name,function(res){
      if((res.type === 12 && pay_status.is_pay && pay_status.advanced !== 'ltd') || !pay_status.is_pay){
        product_recommend.recommend_product_view(item)
      }else if(!res.setup){
        bt.soft.install(item.name)
      }else{
        bt.plugin.get_plugin_byhtml(item.name,function(html){
          if(typeof html === "string"){
            layer.open({ 
              type:1,
              shade:0,
              skin:'hide',
              content:html, 
              success:function(){
                var is_event = false;
                for (var i = 0; i < item.eventList.length; i++) {
                  var data = item.eventList[i];
                  var oldVersion = data.version.replace('.',''),newVersion = res.version.replace('.','');
                  if(newVersion <= oldVersion){
                    is_event = true
                    setTimeout(function () {                                                                                                                                                                                                                                                                                                                                                                                     
                      new Function(data.event.replace('$siteName',param))() 
                    },100)
                    break; 
                  }
                }
                if(!is_event) new Function(item.eventList[item.eventList.length - 1].event.replace('$siteName',param))() 
              }
            })
          }
        })
      }

    })
  },
  /**
   * @description 获取支付状态
   */
  get_pay_status:function(){
    var pro_end = parseInt(bt.get_cookie('pro_end') || -1)
    var ltd_end = parseInt(bt.get_cookie('ltd_end')  || -1)
    var is_pay = pro_end > -1 || ltd_end > -1; // 是否购买付费版本
    var advanced = 'ltd'; // 已购买，企业版优先显示
    if(pro_end === -2 || pro_end > -1) advanced = 'pro';
    if(ltd_end === -2 || ltd_end > -1) advanced = 'ltd';
    var end_time = advanced === 'ltd'? ltd_end:pro_end; // 到期时间
    return { advanced: advanced, is_pay:is_pay,  end_time:end_time };
  },

  pay_product_sign:function (type, source) {
    switch (type) {
      case 'pro':
        bt.soft['updata_' + type](source);
        break;
      case 'ltd':
        bt.soft['updata_' + type](false, source);
        break;
    }
  },
  /**
   * @description 获取项目类型
   * @param {Function} callback 回调函数
   */
  get_product_type:function(callback){
    bt.send('get_pay_type','ajax/get_pay_type',{},function(rdata){
      bt.set_storage('session','get_pay_type',JSON.stringify(rdata))
      if(callback) callback(rdata)
    })
  },
  /**
   * @description 推荐购买产品
   * @param {Object} pay_id 购买的入口id
  */
  recommend_product_view: function (config) {
    var name = config.name.split('_')[0];
    var status = this.get_pay_status();
    bt.open({
      title:false,
      area:'650px',
      btn:false,
      content:'<div class="ptb15" style="display: flex;">\
        <div class="product_view"><img src="/static/images/recommend/'+ name +'.png"/></div>\
        <div class="product_describe ml10">\
          <div class="describe_title">'+ config.pluginName +'</div>\
          <div class="describe_ps">'+ config.ps +'</div>\
          <div class="product_describe_btn">\
            <a class="btn btn-default mr10 btn-sm productPreview '+ (!config.preview?'hide':'') +'" href="'+ config.preview +'" target="_blank">产品预览</a><button class="btn btn-success btn-sm buyNow">立即购买</button>\
          </div>\
        </div>\
      </div>',
      success:function () {
        // 产品预览
        $('.product_view img').click(function () {
          layer.open({
            type:1,
            title:'查看图片',
            area:['650px','450px'],
            closeBtn:2,
            btn:false,
            content:'<img src="/static/images/recommend/'+ name +'.png" style="width:100%" />'
          })
        })
        // 立即购买
        $('.buyNow').click(function(){
          switch (status.advanced) {
            case 'pro':
              bt.soft['updata_' + status.advanced](config.pay);
              break;
            case 'ltd':
              bt.soft['updata_' + status.advanced](false, config.pay);
              break;
          }
        })
      }
    })
  }
}